﻿#include "StdAfx.h"
#include <zmouse.h>
#include "Utils/Task.h"

namespace DuiLib {

    /////////////////////////////////////////////////////////////////////////////////////
    //
    //

    static void GetChildWndRect(HWND hWnd, HWND hChildWnd, RECT &rcChildWnd) {
        ::GetWindowRect(hChildWnd, &rcChildWnd);

        POINT pt;
        pt.x = rcChildWnd.left;
        pt.y = rcChildWnd.top;
        ::ScreenToClient(hWnd, &pt);
        rcChildWnd.left = pt.x;
        rcChildWnd.top = pt.y;

        pt.x = rcChildWnd.right;
        pt.y = rcChildWnd.bottom;
        ::ScreenToClient(hWnd, &pt);
        rcChildWnd.right = pt.x;
        rcChildWnd.bottom = pt.y;
    }

    static UINT MapKeyState() {
        UINT uState = 0;

        if( ::GetKeyState(VK_CONTROL) < 0 ) uState |= MK_CONTROL;

        if( ::GetKeyState(VK_LBUTTON) < 0 ) uState |= MK_LBUTTON;

        if( ::GetKeyState(VK_RBUTTON) < 0 ) uState |= MK_RBUTTON;

        if( ::GetKeyState(VK_SHIFT) < 0 ) uState |= MK_SHIFT;

        if( ::GetKeyState(VK_MENU) < 0 ) uState |= MK_ALT;

        return uState;
    }

    typedef struct tagFINDTABINFO {
        CControlUI *pFocus;
        CControlUI *pLast;
        bool bForward;
        bool bNextIsIt;
    } FINDTABINFO;

    typedef struct tagFINDSHORTCUT {
        TCHAR ch;
        bool bPickNext;
    } FINDSHORTCUT;

    typedef struct tagTIMERINFO {
        CControlUI *pSender;
        UINT nLocalID;
        HWND hWnd;
        UINT uWinTimer;
        bool bKilled;
    } TIMERINFO;


    tagTDrawInfo::tagTDrawInfo() {
        Clear();
    }

    void tagTDrawInfo::Parse(LPCTSTR pStrImage, LPCTSTR pStrModify, CPaintManagerUI *pManager) {
        // 1、aaa.jpg
        // 2、file='aaa.jpg' res='' restype='0' dest='0,0,0,0' source='0,0,0,0' corner='0,0,0,0'
        // mask='#FF0000' fade='255' hole='false' xtiled='false' ytiled='false'
        sDrawString = pStrImage;
        sDrawModify = pStrModify;
        sImageName = pStrImage;

        CDuiString sItem;
        CDuiString sValue;
        LPTSTR pstr = NULL;

        for( int i = 0; i < 2; ++i ) {
            if( i == 1) pStrImage = pStrModify;

            if( !pStrImage ) continue;

            while( *pStrImage != _T('\0') ) {
                sItem.Empty();
                sValue.Empty();

                while( *pStrImage > _T('\0') && *pStrImage <= _T(' ') ) pStrImage = ::CharNext(pStrImage);

                while( *pStrImage != _T('\0') && *pStrImage != _T('=') && *pStrImage > _T(' ') ) {
                    LPTSTR pstrTemp = ::CharNext(pStrImage);

                    while( pStrImage < pstrTemp) {
                        sItem += *pStrImage++;
                    }
                }

                while( *pStrImage > _T('\0') && *pStrImage <= _T(' ') ) pStrImage = ::CharNext(pStrImage);

                if( *pStrImage++ != _T('=') ) break;

                while( *pStrImage > _T('\0') && *pStrImage <= _T(' ') ) pStrImage = ::CharNext(pStrImage);

                if( *pStrImage++ != _T('\'') ) break;

                while( *pStrImage != _T('\0') && *pStrImage != _T('\'') ) {
                    LPTSTR pstrTemp = ::CharNext(pStrImage);

                    while( pStrImage < pstrTemp) {
                        sValue += *pStrImage++;
                    }
                }

                if( *pStrImage++ != _T('\'') ) break;

                if( !sValue.IsEmpty() ) {
                    if( sItem == _T("file") || sItem == _T("res") ) {
                        sImageName = sValue;
                    } else if( sItem == _T("restype") ) {
                        sResType = sValue;
                    } else if( sItem == _T("dest") ) {
                        rcDest.left = _tcstol(sValue, &pstr, 10);
                        ASSERT(pstr);
                        rcDest.top = _tcstol(pstr + 1, &pstr, 10);
                        ASSERT(pstr);
                        rcDest.right = _tcstol(pstr + 1, &pstr, 10);
                        ASSERT(pstr);
                        rcDest.bottom = _tcstol(pstr + 1, &pstr, 10);
                        ASSERT(pstr);
                        // Jeffery: 在解析的时候不进行DPI Scale，下同
                        //pManager->GetDPIObj()->Scale(&rcDest);
                    } else if( sItem == _T("source") ) {
                        rcSource.left = _tcstol(sValue, &pstr, 10);
                        ASSERT(pstr);
                        rcSource.top = _tcstol(pstr + 1, &pstr, 10);
                        ASSERT(pstr);
                        rcSource.right = _tcstol(pstr + 1, &pstr, 10);
                        ASSERT(pstr);
                        rcSource.bottom = _tcstol(pstr + 1, &pstr, 10);
                        ASSERT(pstr);
                        //pManager->GetDPIObj()->Scale(&rcSource);
                    } else if( sItem == _T("corner") ) {
                        rcCorner.left = _tcstol(sValue, &pstr, 10);
                        ASSERT(pstr);
                        rcCorner.top = _tcstol(pstr + 1, &pstr, 10);
                        ASSERT(pstr);
                        rcCorner.right = _tcstol(pstr + 1, &pstr, 10);
                        ASSERT(pstr);
                        rcCorner.bottom = _tcstol(pstr + 1, &pstr, 10);
                        ASSERT(pstr);
                        //pManager->GetDPIObj()->Scale(&rcCorner);
                    } else if( sItem == _T("mask") ) {
                        if( sValue[0] == _T('#'))
                            dwMask = _tcstoul(sValue.GetData() + 1, &pstr, 16);
                        else
                            dwMask = _tcstoul(sValue, &pstr, 16);
                    } else if( sItem == _T("fade") ) {
                        uFade = (BYTE)_tcstoul(sValue, &pstr, 10);
                    } else if( sItem == _T("hole") ) {
                        bHole = (_tcsicmp(sValue, _T("true")) == 0);
                    } else if( sItem == _T("xtiled") ) {
                        bTiledX = (_tcsicmp(sValue, _T("true")) == 0);
                    } else if( sItem == _T("ytiled") ) {
                        bTiledY = (_tcsicmp(sValue, _T("true")) == 0);
                    } else if( sItem == _T("hsl") ) {
                        bHSL = (_tcsicmp(sValue, _T("true")) == 0);
                    }
                    // Jeffery: 是否适应DPI缩放
                    //
                    else if (sItem == _T("adaptdpiscale")) {
                        bAdaptDpiScale = (_tcsicmp(sValue, _T("true")) == 0);
                    }
                }

                if( *pStrImage++ != _T(' ') ) break;
            }
        }

        // 调整DPI资源
        if (bAdaptDpiScale && pManager->GetDPIObj()->GetScale() != 100) {
            // Jeffery: 检查对应DPI的图片是否存在，不存在则使用原图，存在则修改图片名为***@dpi.png格式
            //
            CDuiString newImageName;
            if (!CDPI::AddAPIInfo(sImageName, pManager->GetDPIObj()->GetScale(), newImageName))
                return;

            bool newImageIsExist = false;

            do {
                if (sResType.GetLength() == 0) {
                    CDuiString sFile = CPaintManagerUI::GetResourcePath();

                    // 先尝试直接打开
                    do 
                    {
                      HANDLE hFile = ::CreateFile(
                        newImageName, GENERIC_READ, FILE_SHARE_READ, NULL,
                          OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
                      if (hFile == INVALID_HANDLE_VALUE)
                        break;

                      DWORD dwSize = ::GetFileSize(hFile, NULL);
                      if (dwSize == 0) {
                        CloseHandle(hFile);
                        break;
                      }

                      CloseHandle(hFile);
                      newImageIsExist = true;
                    } while (false);

                    if (newImageIsExist) {
                      break;
                    }

                    if (CPaintManagerUI::GetResourceZip().IsEmpty()) {
                        sFile += newImageName;
                        HANDLE hFile = ::CreateFile(sFile, GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
                        if (hFile == INVALID_HANDLE_VALUE)
                            break;

                        DWORD dwSize = ::GetFileSize(hFile, NULL);
                        if (dwSize == 0) {
                            CloseHandle(hFile);
                            break;
                        }

                        CloseHandle(hFile);
                        newImageIsExist = true;
                    } else {
                        sFile += CPaintManagerUI::GetResourceZip();
                        CDuiString sFilePwd = CPaintManagerUI::GetResourceZipPwd();
                        HZIP hz = NULL;

                        if (CPaintManagerUI::IsCachedResourceZip()) {
                            hz = (HZIP)CPaintManagerUI::GetResourceZipHandle();
                        } else {
#ifdef UNICODE
                            std::string pwd = Unicode2Ansi(sFilePwd.GetData());
                            hz = OpenZip(sFile, pwd.c_str());
#else
                            hz = OpenZip(sFile, sFilePwd);
#endif
                        }

                        if (hz == NULL)
                            break;

                        ZIPENTRY ze;
                        int i = 0;
                        CDuiString key = newImageName;
                        key.Replace(_T("\\"), _T("/"));

                        if (FindZipItem(hz, key, true, &i, &ze) != 0) {
                            if (!CPaintManagerUI::IsCachedResourceZip())
                                CloseZip(hz);
                            break;
                        }

                        DWORD dwSize = ze.unc_size;
                        if (dwSize == 0) {
                            if (!CPaintManagerUI::IsCachedResourceZip())
                                CloseZip(hz);
                            break;
                        }

                        newImageIsExist = true;
                    }
                } else {
                    HINSTANCE dllinstance = CPaintManagerUI::GetResourceDll();
                    HRSRC hResource = ::FindResource(dllinstance, newImageName, sResType);
                    if (hResource == NULL)
                        break;

                    HGLOBAL hGlobal = ::LoadResource(dllinstance, hResource);

                    if (hGlobal == NULL) {
                        FreeResource(hResource);
                        break;
                    }

                    DWORD dwSize = ::SizeofResource(dllinstance, hResource);
                    if (dwSize == 0) {
                        FreeResource(hResource);
                        break;
                    }

                    ::FreeResource(hResource);
                    newImageIsExist = true;
                }
            } while (false);


            if (newImageIsExist) {
                sImageName = newImageName;
            } else {
                bForceOriginImage = true;
            }
        }
    }

    void tagTDrawInfo::Clear() {
        sDrawString.Empty();
        sDrawModify.Empty();
        sImageName.Empty();

        memset(&rcDest, 0, sizeof(RECT));
        memset(&rcSource, 0, sizeof(RECT));
        memset(&rcCorner, 0, sizeof(RECT));
        dwMask = 0;
        uFade = 255;
        bHole = false;
        bTiledX = false;
        bTiledY = false;
        bHSL = false;
        bAdaptDpiScale = true;
        bForceOriginImage = false;
    }

    /////////////////////////////////////////////////////////////////////////////////////
    typedef BOOL (__stdcall *PFUNCUPDATELAYEREDWINDOW)(HWND, HDC, POINT *, SIZE *, HDC, POINT *, COLORREF, BLENDFUNCTION *, DWORD);
    PFUNCUPDATELAYEREDWINDOW g_fUpdateLayeredWindow = NULL;

    HPEN m_hUpdateRectPen = NULL;

    HINSTANCE CPaintManagerUI::m_hResourceInstance = NULL;
    CDuiString CPaintManagerUI::m_pStrResourcePath;
    CDuiString CPaintManagerUI::m_pStrResourceZip;
    CDuiString CPaintManagerUI::m_pStrResourceZipPwd;  //Garfield 20160325 带密码zip包解密
    HANDLE CPaintManagerUI::m_hResourceZip = NULL;
    bool CPaintManagerUI::m_bCachedResourceZip = true;
    UILIB_RESOURCETYPE CPaintManagerUI::m_nResType = UILIB_FILE;
    TResInfo CPaintManagerUI::m_SharedResInfo;
    HINSTANCE CPaintManagerUI::m_hInstance = NULL;
    bool CPaintManagerUI::m_bUseHSL = false;
    short CPaintManagerUI::m_H = 180;
    short CPaintManagerUI::m_S = 100;
    short CPaintManagerUI::m_L = 100;
    CStdPtrArray CPaintManagerUI::m_aPreMessages;
    CStdPtrArray CPaintManagerUI::m_aPlugins;
    DWORD CPaintManagerUI::m_dwUIThreadId = 0;
    std::queue< std::function<void()> > CPaintManagerUI::UIWorks;
    std::mutex CPaintManagerUI::UIWorksMutex;

    CPaintManagerUI::CPaintManagerUI() :
        m_hWndPaint(NULL),
        m_hDcPaint(NULL),
        m_hDcOffscreen(NULL),
        m_hDcBackground(NULL),
        m_bOffscreenPaint(true),
        m_hbmpOffscreen(NULL),
        m_pOffscreenBits(NULL),
        m_hbmpBackground(NULL),
        m_pBackgroundBits(NULL),
        m_hwndTooltip(NULL),
        m_uTimerID(0x1000),
        m_pRoot(NULL),
        m_pFocus(NULL),
        m_pEventHover(NULL),
        m_pEventClick(NULL),
        m_pEventKey(NULL),
        m_bFirstLayout(true),
        m_bFocusNeeded(false),
        m_bUpdateNeeded(false),
        m_bMouseTracking(false),
        m_bMouseCapture(false),
        m_bUsedVirtualWnd(false),
        m_bForceUseSharedRes(false),
        m_nOpacity(0xFF),
        m_bLayered(false),
        m_bLayeredChanged(false),
        m_bShowUpdateRect(false),
        m_bUseGdiplusText(false),
        m_trh(0),
        m_bDragMode(false),
        m_hDragBitmap(NULL),
        m_iHoverTime(400UL),
        m_pDPI(NULL) {
        if (m_SharedResInfo.m_DefaultFontInfo.sFontName.IsEmpty()) {
            m_SharedResInfo.m_dwDefaultDisabledColor = 0xFFA7A6AA;
            m_SharedResInfo.m_dwDefaultFontColor = 0xFF000000;
            m_SharedResInfo.m_dwDefaultLinkFontColor = 0xFF0000FF;
            m_SharedResInfo.m_dwDefaultLinkHoverFontColor = 0xFFD3215F;
            m_SharedResInfo.m_dwDefaultSelectedBkColor = 0xFFBAE4FF;

            LOGFONT lf = { 0 };
            ::GetObject(::GetStockObject(DEFAULT_GUI_FONT), sizeof(LOGFONT), &lf);
            lf.lfCharSet = DEFAULT_CHARSET;
            HFONT hDefaultFont = ::CreateFontIndirect(&lf);
            m_SharedResInfo.m_DefaultFontInfo.hFont = hDefaultFont;
            m_SharedResInfo.m_DefaultFontInfo.sFontName = lf.lfFaceName;
            m_SharedResInfo.m_DefaultFontInfo.iSize = -lf.lfHeight;
            m_SharedResInfo.m_DefaultFontInfo.bBold = (lf.lfWeight >= FW_BOLD);
            m_SharedResInfo.m_DefaultFontInfo.bUnderline = (lf.lfUnderline == TRUE);
            m_SharedResInfo.m_DefaultFontInfo.bItalic = (lf.lfItalic == TRUE);
            ::ZeroMemory(&m_SharedResInfo.m_DefaultFontInfo.tm, sizeof(m_SharedResInfo.m_DefaultFontInfo.tm));
        }

        m_ResInfo.m_dwDefaultDisabledColor = m_SharedResInfo.m_dwDefaultDisabledColor;
        m_ResInfo.m_dwDefaultFontColor = m_SharedResInfo.m_dwDefaultFontColor;
        m_ResInfo.m_dwDefaultLinkFontColor = m_SharedResInfo.m_dwDefaultLinkFontColor;
        m_ResInfo.m_dwDefaultLinkHoverFontColor = m_SharedResInfo.m_dwDefaultLinkHoverFontColor;
        m_ResInfo.m_dwDefaultSelectedBkColor = m_SharedResInfo.m_dwDefaultSelectedBkColor;

        if( m_hUpdateRectPen == NULL ) {
            m_hUpdateRectPen = ::CreatePen(PS_SOLID, 1, RGB(220, 0, 0));
            // Boot Windows Common Controls (for the ToolTip control)
            ::InitCommonControls();
            ::LoadLibrary(_T("msimg32.dll"));
        }

        m_szMinWindow.cx = 0;
        m_szMinWindow.cy = 0;
        m_szMaxWindow.cx = 0;
        m_szMaxWindow.cy = 0;
        m_szInitWindowSize.cx = 0;
        m_szInitWindowSize.cy = 0;
        m_szRoundCorner.cx = m_szRoundCorner.cy = 0;
        ::ZeroMemory(&m_rcSizeBox, sizeof(m_rcSizeBox));
        ::ZeroMemory(&m_rcCaption, sizeof(m_rcCaption));
        ::ZeroMemory(&m_rcLayeredInset, sizeof(m_rcLayeredInset));
        ::ZeroMemory(&m_rcLayeredUpdate, sizeof(m_rcLayeredUpdate));
        m_ptLastMousePos.x = m_ptLastMousePos.y = -1;

        m_pGdiplusStartupInput = new Gdiplus::GdiplusStartupInput;
        Gdiplus::GdiplusStartup( &m_gdiplusToken, m_pGdiplusStartupInput, NULL); // 加载GDI接口

        CShadowUI::Initialize(m_hInstance);
    }

    CPaintManagerUI::~CPaintManagerUI() {
        // Delete the control-tree structures
        for( int i = 0; i < m_aDelayedCleanup.GetSize(); i++ ) delete static_cast<CControlUI *>(m_aDelayedCleanup[i]);

        m_aDelayedCleanup.Resize(0);

        for( int i = 0; i < m_aAsyncNotify.GetSize(); i++ ) delete static_cast<TNotifyUI *>(m_aAsyncNotify[i]);

        m_aAsyncNotify.Resize(0);

        m_mNameHash.RemoveAll();

        if( m_pRoot != NULL ) delete m_pRoot;

        ::DeleteObject(m_ResInfo.m_DefaultFontInfo.hFont);
        RemoveAllFonts();
        RemoveAllImages();
        RemoveAllStyle();
        RemoveAllDefaultAttributeList();
        RemoveAllWindowCustomAttribute();
        RemoveAllOptionGroups();
        RemoveAllTimers();
        RemoveAllDrawInfos();

        if( m_hwndTooltip != NULL ) {
            ::DestroyWindow(m_hwndTooltip);
            m_hwndTooltip = NULL;
        }

        if (!m_aFonts.IsEmpty()) {
            for (int i = 0; i < m_aFonts.GetSize(); ++i) {
                HANDLE handle = static_cast<HANDLE>(m_aFonts.GetAt(i));
                ::RemoveFontMemResourceEx(handle);
            }
        }

        if( m_hDcOffscreen != NULL ) ::DeleteDC(m_hDcOffscreen);

        if( m_hDcBackground != NULL ) ::DeleteDC(m_hDcBackground);

        if( m_hbmpOffscreen != NULL ) ::DeleteObject(m_hbmpOffscreen);

        if( m_hbmpBackground != NULL ) ::DeleteObject(m_hbmpBackground);

        if( m_hDcPaint != NULL ) ::ReleaseDC(m_hWndPaint, m_hDcPaint);

        m_aPreMessages.Remove(m_aPreMessages.Find(this));

        // 销毁拖拽图片
        if( m_hDragBitmap != NULL ) ::DeleteObject(m_hDragBitmap);

        //卸载GDIPlus
        Gdiplus::GdiplusShutdown(m_gdiplusToken);
        delete m_pGdiplusStartupInput;
    }

    void CPaintManagerUI::Init(HWND hWnd, LPCTSTR pstrName) {
        ASSERT(::IsWindow(hWnd));

        RemoveAllFonts();
        RemoveAllImages();
        RemoveAllStyle();
        RemoveAllDefaultAttributeList();
        RemoveAllWindowCustomAttribute();
        RemoveAllOptionGroups();
        RemoveAllTimers();

        m_sName.Empty();

        if( pstrName != NULL ) m_sName = pstrName;

        if( m_hWndPaint != hWnd ) {
            m_hWndPaint = hWnd;
            m_hDcPaint = ::GetDC(hWnd);
            m_aPreMessages.Add(this);
        }
    }

    void CPaintManagerUI::DeletePtr(void *ptr) {
        if(ptr) {
            delete ptr;
            ptr = NULL;
        }
    }

    HINSTANCE CPaintManagerUI::GetInstance() {
        return m_hInstance;
    }

    CDuiString CPaintManagerUI::GetInstancePath() {
        if( m_hInstance == NULL ) return _T('\0');

        TCHAR tszModule[MAX_PATH + 1] = { 0 };
        ::GetModuleFileName(m_hInstance, tszModule, MAX_PATH);
        CDuiString sInstancePath = tszModule;
        int pos = sInstancePath.ReverseFind(_T('\\'));

        if( pos >= 0 ) sInstancePath = sInstancePath.Left(pos + 1);

        return sInstancePath;
    }

    CDuiString CPaintManagerUI::GetCurrentPath() {
        TCHAR tszModule[MAX_PATH + 1] = { 0 };
        ::GetCurrentDirectory(MAX_PATH, tszModule);
        return tszModule;
    }

    HINSTANCE CPaintManagerUI::GetResourceDll() {
        if( m_hResourceInstance == NULL ) return m_hInstance;

        return m_hResourceInstance;
    }

    const CDuiString &CPaintManagerUI::GetResourcePath() {
        return m_pStrResourcePath;
    }

    const CDuiString &CPaintManagerUI::GetResourceZip() {
        return m_pStrResourceZip;
    }

    const CDuiString &CPaintManagerUI::GetResourceZipPwd() {
        return m_pStrResourceZipPwd;
    }

    bool CPaintManagerUI::IsCachedResourceZip() {
        return m_bCachedResourceZip;
    }

    HANDLE CPaintManagerUI::GetResourceZipHandle() {
        return m_hResourceZip;
    }

    void CPaintManagerUI::SetInstance(HINSTANCE hInst) {
        m_hInstance = hInst;
    }

    void CPaintManagerUI::SetCurrentPath(LPCTSTR pStrPath) {
        ::SetCurrentDirectory(pStrPath);
    }

    void CPaintManagerUI::SetResourceDll(HINSTANCE hInst) {
        m_hResourceInstance = hInst;
    }

    void CPaintManagerUI::SetResourcePath(LPCTSTR pStrPath) {
        m_pStrResourcePath = pStrPath;

        if( m_pStrResourcePath.IsEmpty() ) return;

        TCHAR cEnd = m_pStrResourcePath.GetAt(m_pStrResourcePath.GetLength() - 1);

        if( cEnd != _T('\\') && cEnd != _T('/') ) m_pStrResourcePath += _T('\\');
    }

    void CPaintManagerUI::SetResourceZip(LPVOID pVoid, unsigned int len, LPCTSTR password) {
        if( m_pStrResourceZip == _T("membuffer") ) return;

        if( m_bCachedResourceZip && m_hResourceZip != NULL ) {
            CloseZip((HZIP)m_hResourceZip);
            m_hResourceZip = NULL;
        }

        m_pStrResourceZip = _T("membuffer");
        m_bCachedResourceZip = true;
        m_pStrResourceZipPwd = password;  //Garfield 20160325 带密码zip包解密

        if( m_bCachedResourceZip ) {
#ifdef UNICODE
            std::string pwd;

            if(password)
                pwd = Unicode2Ansi(password);

            m_hResourceZip = (HANDLE)OpenZip(pVoid, len, pwd.c_str());
#else
            m_hResourceZip = (HANDLE)OpenZip(pVoid, len, password);
#endif
        }
    }

    void CPaintManagerUI::SetResourceZip(LPCTSTR pStrPath, bool bCachedResourceZip, LPCTSTR password) {
        if( m_pStrResourceZip == pStrPath && m_bCachedResourceZip == bCachedResourceZip ) return;

        if( m_bCachedResourceZip && m_hResourceZip != NULL ) {
            CloseZip((HZIP)m_hResourceZip);
            m_hResourceZip = NULL;
        }

        m_pStrResourceZip = pStrPath;
        m_bCachedResourceZip = bCachedResourceZip;
        m_pStrResourceZipPwd = password;

        if( m_bCachedResourceZip ) {
            CDuiString sFile = CPaintManagerUI::GetResourcePath();
            sFile += CPaintManagerUI::GetResourceZip();
#ifdef UNICODE
            std::string pwd;

            if(password)
                pwd = Unicode2Ansi((wchar_t *)password);

            m_hResourceZip = (HANDLE)OpenZip(sFile, pwd.c_str());
#else
            m_hResourceZip = (HANDLE)OpenZip(sFile, password);
#endif
        }
    }

    void CPaintManagerUI::SetResourceZip(int resourceID, const CDuiString &strResType, LPCTSTR password /*= NULL*/) {
        HRSRC hResource = ::FindResource(CPaintManagerUI::GetResourceDll(), MAKEINTRESOURCE(resourceID), strResType);

        if (hResource != NULL) {
            DWORD dwSize = 0;
            HGLOBAL hGlobal = ::LoadResource(CPaintManagerUI::GetResourceDll(), hResource);

            if (hGlobal != NULL) {
                dwSize = ::SizeofResource(CPaintManagerUI::GetResourceDll(), hResource);

                if (dwSize > 0) {
                    CPaintManagerUI::SetResourceZip((LPBYTE)::LockResource(hGlobal), dwSize, password);
                }
            }

            ::FreeResource(hResource);
        }
    }

    void CPaintManagerUI::SetResourceType(UILIB_RESOURCETYPE nType) {
        m_nResType = nType;
    }

    UILIB_RESOURCETYPE CPaintManagerUI::GetResourceType() {
        return m_nResType;
    }

    bool CPaintManagerUI::GetHSL(short *H, short *S, short *L) {
        *H = m_H;
        *S = m_S;
        *L = m_L;
        return m_bUseHSL;
    }

    void CPaintManagerUI::SetHSL(bool bUseHSL, short H, short S, short L) {
        if( m_bUseHSL || m_bUseHSL != bUseHSL ) {
            m_bUseHSL = bUseHSL;

            if( H == m_H && S == m_S && L == m_L ) return;

            m_H = CLAMP(H, 0, 360);
            m_S = CLAMP(S, 0, 200);
            m_L = CLAMP(L, 0, 200);
            AdjustSharedImagesHSL();

            for( int i = 0; i < m_aPreMessages.GetSize(); i++ ) {
                CPaintManagerUI *pManager = static_cast<CPaintManagerUI *>(m_aPreMessages[i]);

                if( pManager != NULL ) pManager->AdjustImagesHSL();
            }
        }
    }

    void CPaintManagerUI::ReloadSkin() {
        ReloadSharedImages();

        for( int i = 0; i < m_aPreMessages.GetSize(); i++ ) {
            CPaintManagerUI *pManager = static_cast<CPaintManagerUI *>(m_aPreMessages[i]);
            pManager->ReloadImages();
        }
    }

    CPaintManagerUI *CPaintManagerUI::GetPaintManager(LPCTSTR pstrName) {
        if( pstrName == NULL ) return NULL;

        CDuiString sName = pstrName;

        if( sName.IsEmpty() ) return NULL;

        for( int i = 0; i < m_aPreMessages.GetSize(); i++ ) {
            CPaintManagerUI *pManager = static_cast<CPaintManagerUI *>(m_aPreMessages[i]);

            if( pManager != NULL && sName == pManager->GetName() ) return pManager;
        }

        return NULL;
    }

    CStdPtrArray *CPaintManagerUI::GetPaintManagers() {
        return &m_aPreMessages;
    }

    bool CPaintManagerUI::LoadPlugin(LPCTSTR pstrModuleName) {
        ASSERT( !::IsBadStringPtr(pstrModuleName, -1) || pstrModuleName == NULL );

        if( pstrModuleName == NULL ) return false;

        HMODULE hModule = ::LoadLibrary(pstrModuleName);

        if( hModule != NULL ) {
            LPCREATECONTROL lpCreateControl = (LPCREATECONTROL)::GetProcAddress(hModule, "CreateControl");

            if( lpCreateControl != NULL ) {
                if( m_aPlugins.Find(lpCreateControl) >= 0 ) return true;

                m_aPlugins.Add(lpCreateControl);
                return true;
            }
        }

        return false;
    }

    CStdPtrArray *CPaintManagerUI::GetPlugins() {
        return &m_aPlugins;
    }

    DWORD CPaintManagerUI::GetUIThreadId() {
        return m_dwUIThreadId;
    }

    HWND CPaintManagerUI::GetPaintWindow() const {
        return m_hWndPaint;
    }

    HWND CPaintManagerUI::GetTooltipWindow() const {
        return m_hwndTooltip;
    }
    int CPaintManagerUI::GetHoverTime() const {
        return m_iHoverTime;
    }

    void CPaintManagerUI::SetHoverTime(int iTime) {
        m_iHoverTime = iTime;
    }

    LPCTSTR CPaintManagerUI::GetName() const {
        return m_sName;
    }

    HDC CPaintManagerUI::GetPaintDC() const {
        return m_hDcPaint;
    }

    POINT CPaintManagerUI::GetMousePos() const {
        return m_ptLastMousePos;
    }

    SIZE CPaintManagerUI::GetClientSize() const {
        RECT rcClient = { 0 };
        ::GetClientRect(m_hWndPaint, &rcClient);
        return CDuiSize(rcClient.right - rcClient.left, rcClient.bottom - rcClient.top);
    }

    SIZE CPaintManagerUI::GetInitSize() {
        return m_szInitWindowSize;
    }

    void CPaintManagerUI::SetInitSize(int cx, int cy, bool bDpiScale /*= true*/) {
        m_szInitWindowSize.cx = cx;
        m_szInitWindowSize.cy = cy;

        if( m_pRoot == NULL && m_hWndPaint != NULL ) {
            SIZE scaledSize = m_szInitWindowSize;
            if(bDpiScale)
                scaledSize = GetDPIObj()->Scale(m_szInitWindowSize);
            ::SetWindowPos(m_hWndPaint, NULL, 0, 0, scaledSize.cx, scaledSize.cy, SWP_NOZORDER | SWP_NOMOVE | SWP_NOACTIVATE);
        }
    }

    RECT &CPaintManagerUI::GetSizeBox() {
        return m_rcSizeBox;
    }

    void CPaintManagerUI::SetSizeBox(RECT &rcSizeBox) {
        m_rcSizeBox = rcSizeBox;
    }

    RECT &CPaintManagerUI::GetCaptionRect() {
        return m_rcCaption;
    }

    void CPaintManagerUI::SetCaptionRect(RECT &rcCaption) {
        m_rcCaption = rcCaption;
    }

    SIZE CPaintManagerUI::GetRoundCorner() const {
        return m_szRoundCorner;
    }

    void CPaintManagerUI::SetRoundCorner(int cx, int cy) {
        m_szRoundCorner.cx = cx;
        m_szRoundCorner.cy = cy;
    }

    SIZE CPaintManagerUI::GetMinInfo() {
        return GetDPIObj()->Scale(m_szMinWindow);
    }

    void CPaintManagerUI::SetMinInfo(int cx, int cy) {
        ASSERT(cx >= 0 && cy >= 0);
        m_szMinWindow.cx = cx;
        m_szMinWindow.cy = cy;
    }

    SIZE CPaintManagerUI::GetMaxInfo() {
        // Jeffery: 修复maxinfo不支持DPI缩放的问题
        //
        return GetDPIObj()->Scale(m_szMaxWindow);
    }

    void CPaintManagerUI::SetMaxInfo(int cx, int cy) {
        ASSERT(cx >= 0 && cy >= 0);
        m_szMaxWindow.cx = cx;
        m_szMaxWindow.cy = cy;
    }

    bool CPaintManagerUI::IsShowUpdateRect() const {
        return m_bShowUpdateRect;
    }

    void CPaintManagerUI::SetShowUpdateRect(bool show) {
        m_bShowUpdateRect = show;
    }

    bool CPaintManagerUI::IsNoActivate() {
        return m_bNoActivate;
    }

    void CPaintManagerUI::SetNoActivate(bool bNoActivate) {
        m_bNoActivate = bNoActivate;
    }

    BYTE CPaintManagerUI::GetOpacity() const {
        return m_nOpacity;
    }

    void CPaintManagerUI::SetOpacity(BYTE nOpacity) {
        m_nOpacity = nOpacity;

        if( m_hWndPaint != NULL ) {
            typedef BOOL (__stdcall * PFUNCSETLAYEREDWINDOWATTR)(HWND, COLORREF, BYTE, DWORD);
            PFUNCSETLAYEREDWINDOWATTR fSetLayeredWindowAttributes = NULL;

            HMODULE hUser32 = ::GetModuleHandle(_T("User32.dll"));

            if (hUser32) {
                fSetLayeredWindowAttributes =
                    (PFUNCSETLAYEREDWINDOWATTR)::GetProcAddress(hUser32, "SetLayeredWindowAttributes");

                if( fSetLayeredWindowAttributes == NULL ) return;
            }

            DWORD dwStyle = ::GetWindowLong(m_hWndPaint, GWL_EXSTYLE);
            DWORD dwNewStyle = dwStyle;

            if( nOpacity >= 0 && nOpacity < 256 ) dwNewStyle |= WS_EX_LAYERED;
            else dwNewStyle &= ~WS_EX_LAYERED;

            if(dwStyle != dwNewStyle) ::SetWindowLong(m_hWndPaint, GWL_EXSTYLE, dwNewStyle);

            fSetLayeredWindowAttributes(m_hWndPaint, 0, nOpacity, LWA_ALPHA);
        }
    }

    bool CPaintManagerUI::IsLayered() {
        return m_bLayered;
    }

    void CPaintManagerUI::SetLayered(bool bLayered) {
        if( m_hWndPaint != NULL && bLayered != m_bLayered ) {
            UINT uStyle = GetWindowStyle(m_hWndPaint);

            if( (uStyle & WS_CHILD) != 0 ) return;

            if( g_fUpdateLayeredWindow == NULL ) {
                HMODULE hUser32 = ::GetModuleHandle(_T("User32.dll"));

                if (hUser32) {
                    g_fUpdateLayeredWindow =
                        (PFUNCUPDATELAYEREDWINDOW)::GetProcAddress(hUser32, "UpdateLayeredWindow");

                    if( g_fUpdateLayeredWindow == NULL ) return;
                }
            }

            m_bLayered = bLayered;

            if( m_pRoot != NULL ) m_pRoot->NeedUpdate();

            Invalidate();
        }
    }

    RECT &CPaintManagerUI::GetLayeredInset() {
        return m_rcLayeredInset;
    }

    void CPaintManagerUI::SetLayeredInset(RECT &rcLayeredInset) {
        m_rcLayeredInset = rcLayeredInset;
        m_bLayeredChanged = true;
        Invalidate();
    }

    BYTE CPaintManagerUI::GetLayeredOpacity() {
        return m_nOpacity;
    }

    void CPaintManagerUI::SetLayeredOpacity(BYTE nOpacity) {
        m_nOpacity = nOpacity;
        m_bLayeredChanged = true;
        Invalidate();
    }

    LPCTSTR CPaintManagerUI::GetLayeredImage() {
        return m_diLayered.sDrawString;
    }

    void CPaintManagerUI::SetLayeredImage(LPCTSTR pstrImage) {
        m_diLayered.sDrawString = pstrImage;
        RECT rcNull = {0};
        CRenderEngine::DrawImageInfo(NULL, this, rcNull, rcNull, &m_diLayered);
        m_bLayeredChanged = true;
        Invalidate();
    }

    CShadowUI *CPaintManagerUI::GetShadow() {
        return &m_shadow;
    }

    void CPaintManagerUI::SetUseGdiplusText(bool bUse) {
        m_bUseGdiplusText = bUse;
    }

    bool CPaintManagerUI::IsUseGdiplusText() const {
        return m_bUseGdiplusText;
    }

    void CPaintManagerUI::SetGdiplusTextRenderingHint(int trh) {
        m_trh = trh;
    }

    int CPaintManagerUI::GetGdiplusTextRenderingHint() const {
        return m_trh;
    }

    bool CPaintManagerUI::PreMessageHandler(UINT uMsg, WPARAM wParam, LPARAM lParam, LRESULT & /*lRes*/) {
        for( int i = 0; i < m_aPreMessageFilters.GetSize(); i++ ) {
            bool bHandled = false;
            LRESULT lResult = static_cast<IMessageFilterUI *>(m_aPreMessageFilters[i])->MessageHandler(uMsg, wParam, lParam, bHandled);

            if( bHandled ) {
                return true;
            }
        }

        switch( uMsg ) {
            case WM_KEYDOWN: {
                // Tabbing between controls
                if( wParam == VK_TAB ) {
                    if( m_pFocus && m_pFocus->IsVisible() && m_pFocus->IsEnabled() && _tcsstr(m_pFocus->GetClass(), DUI_CTR_RICHEDIT) != NULL ) {
                        if( static_cast<CRichEditUI *>(m_pFocus)->IsWantTab() ) return false;
                    }

                    if( m_pFocus && m_pFocus->IsVisible() && m_pFocus->IsEnabled() && _tcsstr(m_pFocus->GetClass(), DUI_CTR_CEF) != NULL ) {
                        return false;
                    }

                    SetNextTabControl(::GetKeyState(VK_SHIFT) >= 0);
                    return true;
                }
            }
            break;

            case WM_SYSCHAR: {
                if (m_pFocus != NULL) {
                    TEventUI event = { 0 };
                    event.Type = UIEVENT_SYSCHAR;
                    event.chKey = (TCHAR)wParam;
                    event.ptMouse = m_ptLastMousePos;
                    event.wKeyState = MapKeyState();
                    event.dwTimestamp = ::GetTickCount();
                    m_pFocus->Event(event);
                }
                // Handle ALT-shortcut key-combinations
                FINDSHORTCUT fs = { 0 };
                fs.ch = toupper((int)wParam);
                CControlUI *pControl = m_pRoot->FindControl(__FindControlFromShortcut, &fs, UIFIND_ENABLED | UIFIND_ME_FIRST | UIFIND_TOP_FIRST);

                if( pControl != NULL ) {
                    pControl->SetFocus();
                    pControl->Activate();
                    return true;
                }
            }
            break;

            case WM_SYSKEYDOWN: {
                if( m_pFocus != NULL ) {
                    TEventUI event = { 0 };
                    event.Type = UIEVENT_SYSKEYDOWN;
                    event.chKey = (TCHAR)wParam;
                    event.ptMouse = m_ptLastMousePos;
                    event.wKeyState = MapKeyState();
                    event.dwTimestamp = ::GetTickCount();
                    m_pFocus->Event(event);
                }
            }
            break;
            case WM_SYSKEYUP: {
                if (m_pFocus != NULL) {
                    TEventUI event = { 0 };
                    event.Type = UIEVENT_SYSKEYUP;
                    event.chKey = (TCHAR)wParam;
                    event.ptMouse = m_ptLastMousePos;
                    event.wKeyState = MapKeyState();
                    event.dwTimestamp = ::GetTickCount();
                    m_pFocus->Event(event);
                }
            }
            break;
        }

        return false;
    }

    bool CPaintManagerUI::MessageHandler(UINT uMsg, WPARAM wParam, LPARAM lParam, LRESULT &lRes) {
        if( m_hWndPaint == NULL )
            return false;

        // Cycle through listeners
        for( int i = 0; i < m_aMessageFilters.GetSize(); i++ ) {
            bool bHandled = false;
            LRESULT lResult = static_cast<IMessageFilterUI *>(m_aMessageFilters[i])->MessageHandler(uMsg, wParam, lParam, bHandled);

            if( bHandled ) {
                lRes = lResult;

                switch( uMsg ) {
                    case WM_MOUSEMOVE:
                    case WM_LBUTTONDOWN:
                    case WM_LBUTTONDBLCLK:
                    case WM_LBUTTONUP: {
                        POINT pt = { GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam) };
                        m_ptLastMousePos = pt;
                    }
                    break;

                    case WM_CONTEXTMENU:
                    case WM_MOUSEWHEEL: {
                        POINT pt = { GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam) };
                        ::ScreenToClient(m_hWndPaint, &pt);
                        m_ptLastMousePos = pt;
                    }
                    break;
                }

                return true;
            }
        }

        if( m_bLayered ) {
            switch( uMsg ) {
                case WM_NCACTIVATE:
                    if( !::IsIconic(m_hWndPaint) ) {
                        lRes = (wParam == 0) ? TRUE : FALSE;
                        return true;
                    }

                    break;

                case WM_NCCALCSIZE:
                case WM_NCPAINT:
                    lRes = 0;
                    return true;
            }
        }

        // Custom handling of events
        switch( uMsg ) {
            case WM_APP + 1: {
                for( int i = 0; i < m_aDelayedCleanup.GetSize(); i++ )
                    delete static_cast<CControlUI *>(m_aDelayedCleanup[i]);

                m_aDelayedCleanup.Empty();

                m_bAsyncNotifyPosted = false;

                TNotifyUI *pMsg = NULL;

                while( (pMsg = static_cast<TNotifyUI *>(m_aAsyncNotify.GetAt(0))) ) {
                    m_aAsyncNotify.Remove(0);

                    if( pMsg->pSender != NULL ) {
                        if( pMsg->pSender->m_OnNotifyDelegate )
                            pMsg->pSender->m_OnNotifyDelegate(pMsg);
                    }

                    for( int j = 0; j < m_aNotifiers.GetSize(); j++ ) {
                        static_cast<INotifyUI *>(m_aNotifiers[j])->Notify(*pMsg);
                    }

                    delete pMsg;
                }
            }
            break;

            case WM_CLOSE: {
                // Make sure all matching "closing" events are sent
                TEventUI event = { 0 };
                event.ptMouse = m_ptLastMousePos;
                event.wKeyState = MapKeyState();
                event.dwTimestamp = ::GetTickCount();

                if( m_pEventHover != NULL ) {
                    event.Type = UIEVENT_MOUSELEAVE;
                    event.pSender = m_pEventHover;
                    m_pEventHover->Event(event);
                }

                if( m_pEventClick != NULL ) {
                    event.Type = UIEVENT_BUTTONUP;
                    event.pSender = m_pEventClick;
                    m_pEventClick->Event(event);
                }

                SetFocus(NULL);

                if( ::GetActiveWindow() == m_hWndPaint ) {
                    HWND hwndParent = GetWindowOwner(m_hWndPaint);

                    if( hwndParent != NULL )
                        ::SetFocus(hwndParent);
                }

                if (m_hwndTooltip != NULL) {
                    ::DestroyWindow(m_hwndTooltip);
                    m_hwndTooltip = NULL;
                }
            }
            break;

            case WM_ERASEBKGND: {
                // We'll do the painting here...
                lRes = 1;
            }
            return true;

            case WM_PAINT: {
                if( m_pRoot == NULL ) {
                    PAINTSTRUCT ps = { 0 };
                    ::BeginPaint(m_hWndPaint, &ps);
                    CRenderEngine::DrawColor(m_hDcPaint, ps.rcPaint, 0xFF000000);
                    ::EndPaint(m_hWndPaint, &ps);
                    return true;
                }

                RECT rcClient = { 0 };
                ::GetClientRect(m_hWndPaint, &rcClient);

                RECT rcPaint = { 0 };

                if( !::GetUpdateRect(m_hWndPaint, &rcPaint, FALSE) )
                    return true;

                // Set focus to first control?
                if( m_bFocusNeeded ) {
                    SetNextTabControl();
                }

                SetPainting(true);

                bool bNeedSizeMsg = false;
                LONG dwWidth = rcClient.right - rcClient.left;
                LONG dwHeight = rcClient.bottom - rcClient.top;

                SetPainting(true);

                if( m_bUpdateNeeded ) {
                    m_bUpdateNeeded = false;

                    if( !::IsRectEmpty(&rcClient) && !::IsIconic(m_hWndPaint) ) {
                        if( m_pRoot->IsUpdateNeeded() ) {
                            RECT rcRoot = rcClient;

                            if( m_hDcOffscreen != NULL )
                                ::DeleteDC(m_hDcOffscreen);

                            if( m_hDcBackground != NULL )
                                ::DeleteDC(m_hDcBackground);

                            if( m_hbmpOffscreen != NULL )
                                ::DeleteObject(m_hbmpOffscreen);

                            if( m_hbmpBackground != NULL )
                                ::DeleteObject(m_hbmpBackground);

                            m_hDcOffscreen = NULL;
                            m_hDcBackground = NULL;
                            m_hbmpOffscreen = NULL;
                            m_hbmpBackground = NULL;

                            if( m_bLayered ) {
                                rcRoot.left += m_rcLayeredInset.left;
                                rcRoot.top += m_rcLayeredInset.top;
                                rcRoot.right -= m_rcLayeredInset.right;
                                rcRoot.bottom -= m_rcLayeredInset.bottom;
                            }

                            m_pRoot->SetPos(rcRoot, true);
                            bNeedSizeMsg = true;
                        } else {
                            CControlUI *pControl = NULL;
                            m_aFoundControls.Empty();
                            m_pRoot->FindControl(__FindControlsFromUpdate, NULL, UIFIND_VISIBLE | UIFIND_ME_FIRST | UIFIND_UPDATETEST);

                            for( int it = 0; it < m_aFoundControls.GetSize(); it++ ) {
                                pControl = static_cast<CControlUI *>(m_aFoundControls[it]);

                                if( !pControl->IsFloat() )
                                    pControl->SetPos(pControl->GetPos(), true);
                                else
                                    pControl->SetPos(pControl->GetRelativePos(), true);
                            }

                            bNeedSizeMsg = true;
                        }

                        // We'll want to notify the window when it is first initialized
                        // with the correct layout. The window form would take the time
                        // to submit swipes/animations.
                        if( m_bFirstLayout ) {
                            m_bFirstLayout = false;
                            SendNotify(m_pRoot, DUI_MSGTYPE_WINDOWINIT,  0, 0, false);

                            if( m_bLayered && m_bLayeredChanged ) {
                                Invalidate();
                                SetPainting(false);
                                return true;
                            }

                            // 更新阴影窗口显示
                            m_shadow.Update(m_hWndPaint);
                        }
                    }
                } else if( m_bLayered && m_bLayeredChanged ) {
                    RECT rcRoot = rcClient;

                    if( m_pOffscreenBits )
                        ::ZeroMemory(m_pOffscreenBits, (rcRoot.right - rcRoot.left) * (rcRoot.bottom - rcRoot.top) * 4);

                    rcRoot.left += m_rcLayeredInset.left;
                    rcRoot.top += m_rcLayeredInset.top;
                    rcRoot.right -= m_rcLayeredInset.right;
                    rcRoot.bottom -= m_rcLayeredInset.bottom;
                    m_pRoot->SetPos(rcRoot, true);
                }

                if( m_bLayered ) {
                    DWORD dwExStyle = ::GetWindowLong(m_hWndPaint, GWL_EXSTYLE);
                    DWORD dwNewExStyle = dwExStyle | WS_EX_LAYERED;

                    if(dwExStyle != dwNewExStyle)
                        ::SetWindowLong(m_hWndPaint, GWL_EXSTYLE, dwNewExStyle);

                    m_bOffscreenPaint = true;
                    UnionRect(&rcPaint, &rcPaint, &m_rcLayeredUpdate);

                    if( rcPaint.right > rcClient.right )
                        rcPaint.right = rcClient.right;

                    if( rcPaint.bottom > rcClient.bottom )
                        rcPaint.bottom = rcClient.bottom;

                    ::ZeroMemory(&m_rcLayeredUpdate, sizeof(m_rcLayeredUpdate));
                }

                //
                // Render screen
                //
                // Prepare offscreen bitmap
                if( m_bOffscreenPaint && m_hbmpOffscreen == NULL ) {
                    m_hDcOffscreen = ::CreateCompatibleDC(m_hDcPaint);
                    m_hbmpOffscreen = CRenderEngine::CreateARGB32Bitmap(m_hDcPaint, dwWidth, dwHeight, (LPBYTE *)&m_pOffscreenBits);
                    ASSERT(m_hDcOffscreen);
                    ASSERT(m_hbmpOffscreen);
                }

                // Begin Windows paint
                PAINTSTRUCT ps = { 0 };
                ::BeginPaint(m_hWndPaint, &ps);

                if( m_bOffscreenPaint ) {
                    HBITMAP hOldBitmap = (HBITMAP) ::SelectObject(m_hDcOffscreen, m_hbmpOffscreen);
                    int iSaveDC = ::SaveDC(m_hDcOffscreen);

                    if (m_bLayered) {
                        for( LONG y = rcClient.bottom - rcPaint.bottom; y < rcClient.bottom - rcPaint.top; ++y ) {
                            for( LONG x = rcPaint.left; x < rcPaint.right; ++x ) {
                                int i = (y * dwWidth + x) * 4;
                                *(DWORD *)(&m_pOffscreenBits[i]) = 0;
                            }
                        }
                    }

                    m_pRoot->Paint(m_hDcOffscreen, rcPaint, NULL);

                    if( m_bLayered ) {
                        for( int i = 0; i < m_aNativeWindow.GetSize(); ) {
                            HWND hChildWnd = static_cast<HWND>(m_aNativeWindow[i]);

                            if (!::IsWindow(hChildWnd)) {
                                m_aNativeWindow.Remove(i);
                                m_aNativeWindowControl.Remove(i);
                                continue;
                            }

                            ++i;

                            if (!::IsWindowVisible(hChildWnd))
                                continue;

                            RECT rcChildWnd = GetNativeWindowRect(hChildWnd);
                            RECT rcTemp = { 0 };

                            if( !::IntersectRect(&rcTemp, &rcPaint, &rcChildWnd) ) continue;

                            COLORREF *pChildBitmapBits = NULL;
                            HDC hChildMemDC = ::CreateCompatibleDC(m_hDcOffscreen);
                            HBITMAP hChildBitmap = CRenderEngine::CreateARGB32Bitmap(hChildMemDC, rcChildWnd.right - rcChildWnd.left, rcChildWnd.bottom - rcChildWnd.top, (BYTE **)&pChildBitmapBits);
                            ::ZeroMemory(pChildBitmapBits, (rcChildWnd.right - rcChildWnd.left) * (rcChildWnd.bottom - rcChildWnd.top) * 4);
                            HBITMAP hOldChildBitmap = (HBITMAP) ::SelectObject(hChildMemDC, hChildBitmap);
                            ::SendMessage(hChildWnd, WM_PRINT, (WPARAM)hChildMemDC, (LPARAM)(PRF_CHECKVISIBLE | PRF_CHILDREN | PRF_CLIENT | PRF_OWNED));
                            COLORREF *pChildBitmapBit;

                            for( LONG y = 0; y < rcChildWnd.bottom - rcChildWnd.top; y++ ) {
                                for( LONG x = 0; x < rcChildWnd.right - rcChildWnd.left; x++ ) {
                                    pChildBitmapBit = pChildBitmapBits + y * (rcChildWnd.right - rcChildWnd.left) + x;

                                    if (*pChildBitmapBit != 0x00000000)
                                        *pChildBitmapBit |= 0xff000000;
                                }
                            }

                            ::BitBlt(m_hDcOffscreen, rcChildWnd.left, rcChildWnd.top, rcChildWnd.right - rcChildWnd.left,
                                     rcChildWnd.bottom - rcChildWnd.top, hChildMemDC, 0, 0, SRCCOPY);
                            ::SelectObject(hChildMemDC, hOldChildBitmap);
                            ::DeleteObject(hChildBitmap);
                            ::DeleteDC(hChildMemDC);
                        }
                    }

                    for( int i = 0; i < m_aPostPaintControls.GetSize(); i++ ) {
                        CControlUI *pPostPaintControl = static_cast<CControlUI *>(m_aPostPaintControls[i]);
                        pPostPaintControl->DoPostPaint(m_hDcOffscreen, rcPaint);
                    }

                    ::RestoreDC(m_hDcOffscreen, iSaveDC);

                    if( m_bLayered ) {
                        RECT rcWnd = { 0 };
                        ::GetWindowRect(m_hWndPaint, &rcWnd);

                        if(!m_diLayered.sDrawString.IsEmpty()) {
                            DWORD dwWidth1 = rcClient.right - rcClient.left;
                            DWORD dwHeight1 = rcClient.bottom - rcClient.top;
                            RECT rcLayeredClient = rcClient;
                            rcLayeredClient.left += m_rcLayeredInset.left;
                            rcLayeredClient.top += m_rcLayeredInset.top;
                            rcLayeredClient.right -= m_rcLayeredInset.right;
                            rcLayeredClient.bottom -= m_rcLayeredInset.bottom;

                            COLORREF *pOffscreenBits = (COLORREF *)m_pOffscreenBits;
                            COLORREF *pBackgroundBits = m_pBackgroundBits;
                            BYTE A = 0;
                            BYTE R = 0;
                            BYTE G = 0;
                            BYTE B = 0;

                            if (!m_diLayered.sDrawString.IsEmpty()) {
                                if( m_hbmpBackground == NULL) {
                                    m_hDcBackground = ::CreateCompatibleDC(m_hDcPaint);
                                    m_hbmpBackground = CRenderEngine::CreateARGB32Bitmap(m_hDcPaint, dwWidth1, dwHeight1, (BYTE **)&m_pBackgroundBits);
                                    ::ZeroMemory(m_pBackgroundBits, dwWidth1 * dwHeight1 * 4);
                                    ::SelectObject(m_hDcBackground, m_hbmpBackground);
                                    CRenderClip clip;
                                    CRenderClip::GenerateClip(m_hDcBackground, rcLayeredClient, clip);
                                    CRenderEngine::DrawImageInfo(m_hDcBackground, this, rcLayeredClient, rcLayeredClient, &m_diLayered);
                                } else if( m_bLayeredChanged ) {
                                    ::ZeroMemory(m_pBackgroundBits, dwWidth1 * dwHeight1 * 4);
                                    CRenderClip clip;
                                    CRenderClip::GenerateClip(m_hDcBackground, rcLayeredClient, clip);
                                    CRenderEngine::DrawImageInfo(m_hDcBackground, this, rcLayeredClient, rcLayeredClient, &m_diLayered);
                                }

                                for( LONG y = rcClient.bottom - rcPaint.bottom; y < rcClient.bottom - rcPaint.top; ++y ) {
                                    for( LONG x = rcPaint.left; x < rcPaint.right; ++x ) {
                                        pOffscreenBits = (COLORREF *)(m_pOffscreenBits + y * dwWidth1 + x);
                                        pBackgroundBits = m_pBackgroundBits + y * dwWidth1 + x;
                                        A = (BYTE)((*pBackgroundBits) >> 24);
                                        R = (BYTE)((*pOffscreenBits) >> 16) * A / 255;
                                        G = (BYTE)((*pOffscreenBits) >> 8) * A / 255;
                                        B = (BYTE)(*pOffscreenBits) * A / 255;
                                        *pOffscreenBits = RGB(B, G, R) + ((DWORD)A << 24);
                                    }
                                }
                            }
                        } else {
                            for( LONG y = rcClient.bottom - rcPaint.bottom; y < rcClient.bottom - rcPaint.top; ++y ) {
                                for( LONG x = rcPaint.left; x < rcPaint.right; ++x ) {
                                    int i = (y * dwWidth + x) * 4;

                                    if((m_pOffscreenBits[i + 3] == 0) && (m_pOffscreenBits[i + 0] != 0 || m_pOffscreenBits[i + 1] != 0 || m_pOffscreenBits[i + 2] != 0))
                                        m_pOffscreenBits[i + 3] = 255;
                                }
                            }
                        }

                        BLENDFUNCTION bf = { AC_SRC_OVER, 0, m_nOpacity, AC_SRC_ALPHA };
                        POINT ptPos   = { rcWnd.left, rcWnd.top };
                        SIZE sizeWnd  = { dwWidth, dwHeight };
                        POINT ptSrc   = { 0, 0 };
                        g_fUpdateLayeredWindow(m_hWndPaint, m_hDcPaint, &ptPos, &sizeWnd, m_hDcOffscreen, &ptSrc, 0, &bf, ULW_ALPHA);
                    } else {
                        ::BitBlt(m_hDcPaint, rcPaint.left, rcPaint.top, rcPaint.right - rcPaint.left, rcPaint.bottom - rcPaint.top, m_hDcOffscreen, rcPaint.left, rcPaint.top, SRCCOPY);
                    }

                    ::SelectObject(m_hDcOffscreen, hOldBitmap);

                    if( m_bShowUpdateRect && !m_bLayered ) {
                        HPEN hOldPen = (HPEN)::SelectObject(m_hDcPaint, m_hUpdateRectPen);
                        ::SelectObject(m_hDcPaint, ::GetStockObject(HOLLOW_BRUSH));
                        ::Rectangle(m_hDcPaint, rcPaint.left, rcPaint.top, rcPaint.right, rcPaint.bottom);
                        ::SelectObject(m_hDcPaint, hOldPen);
                    }
                } else {
                    // A standard paint job
                    int iSaveDC = ::SaveDC(m_hDcPaint);
                    m_pRoot->Paint(m_hDcPaint, rcPaint, NULL);

                    for( int i = 0; i < m_aPostPaintControls.GetSize(); i++ ) {
                        CControlUI *pPostPaintControl = static_cast<CControlUI *>(m_aPostPaintControls[i]);
                        pPostPaintControl->DoPostPaint(m_hDcPaint, rcPaint);
                    }

                    ::RestoreDC(m_hDcPaint, iSaveDC);
                }

                // All Done!
                ::EndPaint(m_hWndPaint, &ps);

                // 绘制结束
                SetPainting(false);
                m_bLayeredChanged = false;

                if( m_bUpdateNeeded )
                    Invalidate();

                // 发送窗口大小改变消息
                if(bNeedSizeMsg) {
                    this->SendNotify(m_pRoot, DUI_MSGTYPE_WINDOWSIZE, 0, 0, true);
                }

                return true;
            }

            case WM_PRINTCLIENT: {
                if( m_pRoot == NULL ) break;

                RECT rcClient;
                ::GetClientRect(m_hWndPaint, &rcClient);
                HDC hDC = (HDC) wParam;
                int save = ::SaveDC(hDC);
                m_pRoot->Paint(hDC, rcClient, NULL);

                if( (lParam & PRF_CHILDREN) != 0 ) {
                    HWND hWndChild = ::GetWindow(m_hWndPaint, GW_CHILD);

                    while( hWndChild != NULL ) {
                        RECT rcPos = { 0 };
                        ::GetWindowRect(hWndChild, &rcPos);
                        ::MapWindowPoints(HWND_DESKTOP, m_hWndPaint, reinterpret_cast<LPPOINT>(&rcPos), 2);
                        ::SetWindowOrgEx(hDC, -rcPos.left, -rcPos.top, NULL);
                        ::SendMessage(hWndChild, WM_PRINT, wParam, lParam | PRF_NONCLIENT);
                        hWndChild = ::GetWindow(hWndChild, GW_HWNDNEXT);
                    }
                }

                ::RestoreDC(hDC, save);
            }
            break;

            case WM_GETMINMAXINFO: {
                MONITORINFO Monitor = {};
                Monitor.cbSize = sizeof(Monitor);
                ::GetMonitorInfo(::MonitorFromWindow(m_hWndPaint, MONITOR_DEFAULTTOPRIMARY), &Monitor);
                RECT rcWork = Monitor.rcWork;

                if( Monitor.dwFlags != MONITORINFOF_PRIMARY ) {
                    ::OffsetRect(&rcWork, -rcWork.left, -rcWork.top);
                }

                LPMINMAXINFO lpMMI = (LPMINMAXINFO) lParam;

                if( m_szMinWindow.cx > 0 ) lpMMI->ptMinTrackSize.x = m_szMinWindow.cx;

                if( m_szMinWindow.cy > 0 ) lpMMI->ptMinTrackSize.y = m_szMinWindow.cy;

                if( m_szMaxWindow.cx > 0 ) lpMMI->ptMaxTrackSize.x = m_szMaxWindow.cx;

                if( m_szMaxWindow.cy > 0 ) lpMMI->ptMaxTrackSize.y = m_szMaxWindow.cy;

                if( m_szMaxWindow.cx > 0 ) lpMMI->ptMaxSize.x = m_szMaxWindow.cx;

                if( m_szMaxWindow.cy > 0 ) lpMMI->ptMaxSize.y = m_szMaxWindow.cy;
            }
            break;

            case WM_SIZE: {
                if( m_pFocus != NULL ) {
                    TEventUI event = { 0 };
                    event.Type = UIEVENT_WINDOWSIZE;
                    event.pSender = m_pFocus;
                    event.dwTimestamp = ::GetTickCount();
                    m_pFocus->Event(event);
                }

                if( m_pRoot != NULL ) m_pRoot->NeedUpdate();
            }

            return true;

            case WM_TIMER: {
                for( int i = 0; i < m_aTimers.GetSize(); i++ ) {
                    const TIMERINFO *pTimer = static_cast<TIMERINFO *>(m_aTimers[i]);

                    if(pTimer->hWnd == m_hWndPaint &&
                            pTimer->uWinTimer == LOWORD(wParam) &&
                            pTimer->bKilled == false) {
                        TEventUI event = { 0 };
                        event.Type = UIEVENT_TIMER;
                        event.pSender = pTimer->pSender;
                        event.dwTimestamp = ::GetTickCount();
                        event.ptMouse = m_ptLastMousePos;
                        event.wKeyState = MapKeyState();
                        event.wParam = pTimer->nLocalID;
                        event.lParam = lParam;
                        pTimer->pSender->Event(event);
                        break;
                    }
                }
            }
            break;

            case WM_MOUSEHOVER: {
                m_bMouseTracking = false;
                POINT pt = { GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam) };
                CControlUI *pHover = FindControl(pt);

                if( pHover == NULL ) break;

                // Generate mouse hover event
                if( m_pEventHover != NULL ) {
                    TEventUI event = { 0 };
                    event.Type = UIEVENT_MOUSEHOVER;
                    event.pSender = m_pEventHover;
                    event.wParam = wParam;
                    event.lParam = lParam;
                    event.dwTimestamp = ::GetTickCount();
                    event.ptMouse = pt;
                    event.wKeyState = MapKeyState();
                    m_pEventHover->Event(event);
                }

                // Create tooltip information
                CDuiString sToolTip = pHover->GetToolTip();
                if( !sToolTip.IsEmpty() )
                    ShowToolTip(sToolTip.GetData(), { pHover->GetPos().left, pHover->GetPos().top });
            }

            return true;

            case WM_MOUSELEAVE: {
                HideToolTip();

                if( m_bMouseTracking ) {
                    POINT pt = { 0 };
                    RECT rcWnd = { 0 };
                    ::GetCursorPos(&pt);
                    ::GetWindowRect(m_hWndPaint, &rcWnd);

                    if( !::IsIconic(m_hWndPaint) && ::GetActiveWindow() == m_hWndPaint && ::PtInRect(&rcWnd, pt) ) {
                        if( ::SendMessage(m_hWndPaint, WM_NCHITTEST, 0, MAKELPARAM(pt.x, pt.y)) == HTCLIENT ) {
                            ::ScreenToClient(m_hWndPaint, &pt);
                            ::SendMessage(m_hWndPaint, WM_MOUSEMOVE, 0, MAKELPARAM(pt.x, pt.y));
                        } else
                            ::SendMessage(m_hWndPaint, WM_MOUSEMOVE, 0, (LPARAM) - 1);
                    } else
                        ::SendMessage(m_hWndPaint, WM_MOUSEMOVE, 0, (LPARAM) - 1);
                }

                m_bMouseTracking = false;
            }
            break;

            case WM_MOUSEMOVE: {
                // Start tracking this entire window again...
                if( !m_bMouseTracking ) {
                    TRACKMOUSEEVENT tme = { 0 };
                    tme.cbSize = sizeof(TRACKMOUSEEVENT);
                    tme.dwFlags = TME_HOVER | TME_LEAVE;
                    tme.hwndTrack = m_hWndPaint;
                    tme.dwHoverTime = m_hwndTooltip == NULL ? m_iHoverTime : (DWORD) ::SendMessage(m_hwndTooltip, TTM_GETDELAYTIME, TTDT_INITIAL, 0L);
                    _TrackMouseEvent(&tme);
                    m_bMouseTracking = true;
                }

                // Generate the appropriate mouse messages
                POINT pt = { GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam) };
                // 是否移动
                bool bNeedDrag = true;

                if(m_ptLastMousePos.x == pt.x && m_ptLastMousePos.y == pt.y) {
                    bNeedDrag = false;
                }

                // 记录鼠标位置
                m_ptLastMousePos = pt;
                CControlUI *pNewHover = FindControl(pt);

                if( pNewHover != NULL && pNewHover->GetManager() != this ) break;

                TEventUI event = { 0 };
                event.ptMouse = pt;
                event.wParam = wParam;
                event.lParam = lParam;
                event.dwTimestamp = ::GetTickCount();
                event.wKeyState = MapKeyState();

                if( !IsCaptured() ) {
                    pNewHover = FindControl(pt);

                    if( pNewHover != NULL && pNewHover->GetManager() != this ) break;

                    if( pNewHover != m_pEventHover && m_pEventHover != NULL ) {
                        event.Type = UIEVENT_MOUSELEAVE;
                        event.pSender = m_pEventHover;

                        CStdPtrArray aNeedMouseLeaveNeeded(m_aNeedMouseLeaveNeeded.GetSize());
                        aNeedMouseLeaveNeeded.Resize(m_aNeedMouseLeaveNeeded.GetSize());
                        ::CopyMemory(aNeedMouseLeaveNeeded.GetData(), m_aNeedMouseLeaveNeeded.GetData(), m_aNeedMouseLeaveNeeded.GetSize() * sizeof(LPVOID));

                        for( int i = 0; i < aNeedMouseLeaveNeeded.GetSize(); i++ ) {
                            static_cast<CControlUI *>(aNeedMouseLeaveNeeded[i])->Event(event);
                        }

                        m_pEventHover->Event(event);
                        m_pEventHover = NULL;

                        if( m_hwndTooltip != NULL ) ::SendMessage(m_hwndTooltip, TTM_TRACKACTIVATE, FALSE, (LPARAM) &m_ToolTip);
                    }

                    if( pNewHover != m_pEventHover && pNewHover != NULL ) {
                        event.Type = UIEVENT_MOUSEENTER;
                        event.pSender = pNewHover;
                        pNewHover->Event(event);
                        m_pEventHover = pNewHover;
                    }
                }

                if( m_pEventClick != NULL ) {
                    event.Type = UIEVENT_MOUSEMOVE;
                    event.pSender = m_pEventClick;
                    m_pEventClick->Event(event);
                } else if( pNewHover != NULL ) {
                    event.Type = UIEVENT_MOUSEMOVE;
                    event.pSender = pNewHover;
                    pNewHover->Event(event);
                }
            }
            break;

            case WM_LBUTTONDOWN: {
                // We alway set focus back to our app (this helps
                // when Win32 child windows are placed on the dialog
                // and we need to remove them on focus change).
                if (!m_bNoActivate) ::SetFocus(m_hWndPaint);

                if( m_pRoot == NULL ) break;

                // 查找控件
                POINT pt = { GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam) };
                m_ptLastMousePos = pt;
                CControlUI *pControl = FindControl(pt);

                if( pControl == NULL ) break;

                if( pControl->GetManager() != this ) break;

                // 准备拖拽
                if(pControl->IsDragEnabled()) {
                    m_bDragMode = true;

                    if( m_hDragBitmap != NULL ) {
                        ::DeleteObject(m_hDragBitmap);
                        m_hDragBitmap = NULL;
                    }

                    m_hDragBitmap = CRenderEngine::GenerateBitmap(this, pControl, pControl->GetPos());
                }

                // 开启捕获
                SetCapture();
                // 事件处理
                m_pEventClick = pControl;
                pControl->SetFocus();

                TEventUI event = { 0 };
                event.Type = UIEVENT_BUTTONDOWN;
                event.pSender = pControl;
                event.wParam = wParam;
                event.lParam = lParam;
                event.ptMouse = pt;
                event.wKeyState = (WORD)wParam;
                event.dwTimestamp = ::GetTickCount();
                pControl->Event(event);
            }
            break;

            case WM_LBUTTONDBLCLK: {
                if (!m_bNoActivate) ::SetFocus(m_hWndPaint);

                POINT pt = { GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam) };
                m_ptLastMousePos = pt;
                CControlUI *pControl = FindControl(pt);

                if( pControl == NULL )
                    break;

                if( pControl->GetManager() != this )
                    break;

                SetCapture();
                TEventUI event = { 0 };
                event.Type = UIEVENT_DBLCLICK;
                event.pSender = pControl;
                event.ptMouse = pt;
                event.wParam = wParam;
                event.lParam = lParam;
                event.wKeyState = (WORD)wParam;
                event.dwTimestamp = ::GetTickCount();
                pControl->Event(event);
                m_pEventClick = pControl;
            }
            break;

            case WM_LBUTTONUP: {
                POINT pt = { GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam) };
                m_ptLastMousePos = pt;

                if( m_pEventClick == NULL )
                    break;

                ReleaseCapture();
                TEventUI event = { 0 };
                event.Type = UIEVENT_BUTTONUP;
                event.pSender = m_pEventClick;
                event.wParam = wParam;
                event.lParam = lParam;
                event.ptMouse = pt;
                event.wKeyState = (WORD)wParam;
                event.dwTimestamp = ::GetTickCount();

                CControlUI *pClick = m_pEventClick;
                m_pEventClick = NULL;
                pClick->Event(event);
            }
            break;

            case WM_RBUTTONDOWN: {
                if (!m_bNoActivate) ::SetFocus(m_hWndPaint);

                POINT pt = { GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam) };
                m_ptLastMousePos = pt;
                CControlUI *pControl = FindControl(pt);

                if( pControl == NULL )
                    break;

                if( pControl->GetManager() != this )
                    break;

                pControl->SetFocus();
                SetCapture();
                TEventUI event = { 0 };
                event.Type = UIEVENT_RBUTTONDOWN;
                event.pSender = pControl;
                event.wParam = wParam;
                event.lParam = lParam;
                event.ptMouse = pt;
                event.wKeyState = (WORD)wParam;
                event.dwTimestamp = ::GetTickCount();
                pControl->Event(event);
                m_pEventClick = pControl;
            }
            break;

            case WM_RBUTTONUP: {
                POINT pt = { GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam) };
                m_ptLastMousePos = pt;
                m_pEventClick = FindControl(pt);

                if(m_pEventClick == NULL) break;

                ReleaseCapture();
                TEventUI event = { 0 };
                event.Type = UIEVENT_RBUTTONUP;
                event.pSender = m_pEventClick;
                event.wParam = wParam;
                event.lParam = lParam;
                event.ptMouse = pt;
                event.wKeyState = (WORD)wParam;
                event.dwTimestamp = ::GetTickCount();
                m_pEventClick->Event(event);
            }
            break;

            case WM_MBUTTONDOWN: {
                if (!m_bNoActivate) ::SetFocus(m_hWndPaint);

                POINT pt = { GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam) };
                m_ptLastMousePos = pt;
                CControlUI *pControl = FindControl(pt);

                if( pControl == NULL ) break;

                if( pControl->GetManager() != this )
                    break;

                pControl->SetFocus();
                SetCapture();
                TEventUI event = { 0 };
                event.Type = UIEVENT_MBUTTONDOWN;
                event.pSender = pControl;
                event.wParam = wParam;
                event.lParam = lParam;
                event.ptMouse = pt;
                event.wKeyState = (WORD)wParam;
                event.dwTimestamp = ::GetTickCount();
                pControl->Event(event);
                m_pEventClick = pControl;
            }
            break;

            case WM_MBUTTONUP: {
                POINT pt = { GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam) };
                m_ptLastMousePos = pt;
                m_pEventClick = FindControl(pt);

                if(m_pEventClick == NULL) break;

                ReleaseCapture();

                TEventUI event = { 0 };
                event.Type = UIEVENT_MBUTTONUP;
                event.pSender = m_pEventClick;
                event.wParam = wParam;
                event.lParam = lParam;
                event.ptMouse = pt;
                event.wKeyState = (WORD)wParam;
                event.dwTimestamp = ::GetTickCount();
                m_pEventClick->Event(event);
            }
            break;

            case WM_CONTEXTMENU: {
                if( m_pRoot == NULL ) break;

                POINT pt = { GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam) };
                ::ScreenToClient(m_hWndPaint, &pt);
                m_ptLastMousePos = pt;

                if( m_pEventClick == NULL ) break;

                ReleaseCapture();
                TEventUI event = { 0 };
                event.Type = UIEVENT_CONTEXTMENU;
                event.pSender = m_pEventClick;
                event.wParam = wParam;
                event.lParam = lParam;
                event.ptMouse = pt;
                event.wKeyState = (WORD)wParam;
                event.lParam = (LPARAM)m_pEventClick;
                event.dwTimestamp = ::GetTickCount();
                m_pEventClick->Event(event);
                m_pEventClick = NULL;
            }
            break;

            case WM_MOUSEWHEEL: {
                if( m_pRoot == NULL ) break;

                POINT pt = { GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam) };
                ::ScreenToClient(m_hWndPaint, &pt);
                m_ptLastMousePos = pt;
                CControlUI *pControl = FindControl(pt);

                if( pControl == NULL ) break;

                if( pControl->GetManager() != this ) break;

                TEventUI event = { 0 };
                event.Type = UIEVENT_SCROLLWHEEL;
                event.pSender = pControl;
                event.wParam = wParam;
                event.lParam = lParam;
                event.ptMouse = pt;
                event.wKeyState = MapKeyState();
                event.dwTimestamp = ::GetTickCount();
                pControl->Event(event);

                // Let's make sure that the scroll item below the cursor is the same as before...
                ::SendMessage(m_hWndPaint, WM_MOUSEMOVE, 0, (LPARAM) MAKELPARAM(m_ptLastMousePos.x, m_ptLastMousePos.y));
            }
            break;

            case WM_CHAR: {
                if( m_pRoot == NULL ) break;

                if( m_pFocus == NULL ) break;

                TEventUI event = { 0 };
                event.Type = UIEVENT_CHAR;
                event.pSender = m_pFocus;
                event.wParam = wParam;
                event.lParam = lParam;
                event.chKey = (TCHAR)wParam;
                event.ptMouse = m_ptLastMousePos;
                event.wKeyState = MapKeyState();
                event.dwTimestamp = ::GetTickCount();
                m_pFocus->Event(event);
            }
            break;

            case WM_KEYDOWN: {
                if( m_pRoot == NULL ) break;

                if( m_pFocus == NULL ) break;

                TEventUI event = { 0 };
                event.Type = UIEVENT_KEYDOWN;
                event.pSender = m_pFocus;
                event.wParam = wParam;
                event.lParam = lParam;
                event.chKey = (TCHAR)wParam;
                event.ptMouse = m_ptLastMousePos;
                event.wKeyState = MapKeyState();
                event.dwTimestamp = ::GetTickCount();
                m_pFocus->Event(event);
                m_pEventKey = m_pFocus;
            }
            break;

            case WM_KEYUP: {
                if( m_pRoot == NULL ) break;

                if( m_pEventKey == NULL ) break;

                TEventUI event = { 0 };
                event.Type = UIEVENT_KEYUP;
                event.pSender = m_pEventKey;
                event.wParam = wParam;
                event.lParam = lParam;
                event.chKey = (TCHAR)wParam;
                event.ptMouse = m_ptLastMousePos;
                event.wKeyState = MapKeyState();
                event.dwTimestamp = ::GetTickCount();
                m_pEventKey->Event(event);
                m_pEventKey = NULL;
            }
            break;

            case WM_SETCURSOR: {
                if( m_pRoot == NULL ) break;

                if( LOWORD(lParam) != HTCLIENT ) break;

                if( m_bMouseCapture ) return true;

                POINT pt = { 0 };
                ::GetCursorPos(&pt);
                ::ScreenToClient(m_hWndPaint, &pt);
                CControlUI *pControl = FindControl(pt);

                if( pControl == NULL ) break;

                if( (pControl->GetControlFlags() & UIFLAG_SETCURSOR) == 0 ) break;

                TEventUI event = { 0 };
                event.Type = UIEVENT_SETCURSOR;
                event.pSender = pControl;
                event.wParam = wParam;
                event.lParam = lParam;
                event.ptMouse = pt;
                event.wKeyState = MapKeyState();
                event.dwTimestamp = ::GetTickCount();
                pControl->Event(event);
            }

            return true;

            case WM_SETFOCUS: {
                if( m_pFocus != NULL ) {
                    TEventUI event = { 0 };
                    event.Type = UIEVENT_SETFOCUS;
                    event.wParam = wParam;
                    event.lParam = lParam;
                    event.pSender = m_pFocus;
                    event.dwTimestamp = ::GetTickCount();
                    m_pFocus->Event(event);
                }

                break;
            }

            case WM_KILLFOCUS: {
                if(IsCaptured()) ReleaseCapture();

                break;
            }

            case WM_NOTIFY: {
                if( lParam == 0 ) break;

                LPNMHDR lpNMHDR = (LPNMHDR) lParam;

                if( lpNMHDR != NULL ) lRes = ::SendMessage(lpNMHDR->hwndFrom, OCM__BASE + uMsg, wParam, lParam);

                return true;
            }
            break;

            case WM_COMMAND: {
                if( lParam == 0 ) break;

                HWND hWndChild = (HWND) lParam;
                lRes = ::SendMessage(hWndChild, OCM__BASE + uMsg, wParam, lParam);

                if(lRes != 0) return true;
            }
            break;

            case WM_CTLCOLOREDIT:
            case WM_CTLCOLORSTATIC: {
                // Refer To: http://msdn.microsoft.com/en-us/library/bb761691(v=vs.85).aspx
                // Read-only or disabled edit controls do not send the WM_CTLCOLOREDIT message; instead, they send the WM_CTLCOLORSTATIC message.
                if( lParam == 0 ) break;

                HWND hWndChild = (HWND) lParam;
                lRes = ::SendMessage(hWndChild, OCM__BASE + uMsg, wParam, lParam);

                if(lRes != 0) return true;
            }
            break;

            default:
                break;
        }

        return false;
    }

    void CPaintManagerUI::ShowToolTip(LPCTSTR pszToolTip, POINT point) {
        ::ZeroMemory(&m_ToolTip, sizeof(TOOLINFO));
        m_ToolTip.cbSize = sizeof(TOOLINFO);
        m_ToolTip.uFlags = TTF_IDISHWND;
        m_ToolTip.hwnd = m_hWndPaint;
        m_ToolTip.uId = (UINT_PTR)m_hWndPaint;
        m_ToolTip.hinst = m_hInstance;
        m_ToolTip.lpszText = const_cast<LPTSTR>((LPCTSTR)pszToolTip);
        m_ToolTip.rect = { point.x, point.y, 0, 0 };

        if (m_hwndTooltip == NULL) {
            m_hwndTooltip = ::CreateWindowEx(0, TOOLTIPS_CLASS, NULL, WS_POPUP | TTS_NOPREFIX | TTS_ALWAYSTIP, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, m_hWndPaint, NULL, m_hInstance, NULL);
            ::SendMessage(m_hwndTooltip, TTM_ADDTOOL, 0, (LPARAM)&m_ToolTip);
            ::SendMessage(m_hwndTooltip, TTM_SETMAXTIPWIDTH, 0, 300);
        }

        if (!::IsWindowVisible(m_hwndTooltip)) {
            ::SendMessage(m_hwndTooltip, TTM_SETTOOLINFO, 0, (LPARAM)&m_ToolTip);
            ::SendMessage(m_hwndTooltip, TTM_TRACKACTIVATE, TRUE, (LPARAM)&m_ToolTip);
        }
    }

    void CPaintManagerUI::HideToolTip() {
        if (m_hwndTooltip)
            ::SendMessage(m_hwndTooltip, TTM_TRACKACTIVATE, FALSE, (LPARAM)&m_ToolTip);
    }

    bool CPaintManagerUI::IsUpdateNeeded() const {
        return m_bUpdateNeeded;
    }

    void CPaintManagerUI::NeedUpdate() {
        m_bUpdateNeeded = true;
    }

    void CPaintManagerUI::Invalidate() {
        RECT rcClient = { 0 };
        ::GetClientRect(m_hWndPaint, &rcClient);
        ::UnionRect(&m_rcLayeredUpdate, &m_rcLayeredUpdate, &rcClient);
        ::InvalidateRect(m_hWndPaint, NULL, FALSE);
    }

    void CPaintManagerUI::Invalidate(RECT &rcItem) {
        if( rcItem.left < 0 ) rcItem.left = 0;

        if( rcItem .top < 0 ) rcItem.top = 0;

        if( rcItem.right < rcItem.left ) rcItem.right = rcItem.left;

        if( rcItem.bottom < rcItem.top ) rcItem.bottom = rcItem.top;

        ::UnionRect(&m_rcLayeredUpdate, &m_rcLayeredUpdate, &rcItem);
        ::InvalidateRect(m_hWndPaint, &rcItem, FALSE);
    }

    bool CPaintManagerUI::AttachDialog(CControlUI *pControl) {
        ASSERT(::IsWindow(m_hWndPaint));
        // 创建阴影窗口
        m_shadow.Create(this);

        // Reset any previous attachment
        SetFocus(NULL);
        m_pEventKey = NULL;
        m_pEventHover = NULL;
        m_pEventClick = NULL;

        // Remove the existing control-tree. We might have gotten inside this function as
        // a result of an event fired or similar, so we cannot just delete the objects and
        // pull the internal memory of the calling code. We'll delay the cleanup.
        if( m_pRoot != NULL ) {
            m_aPostPaintControls.Empty();
            AddDelayedCleanup(m_pRoot);
        }

        // Set the dialog root element
        m_pRoot = pControl;
        // Go ahead...
        m_bUpdateNeeded = true;
        m_bFirstLayout = true;
        m_bFocusNeeded = true;
        // Initiate all control
        return InitControls(pControl);
    }

    bool CPaintManagerUI::InitControls(CControlUI *pControl, CControlUI *pParent /*= NULL*/) {
        ASSERT(pControl);

        if( pControl == NULL ) return false;

        pControl->SetManager(this, pParent != NULL ? pParent : pControl->GetParent(), true);
        pControl->FindControl(__FindControlFromNameHash, this, UIFIND_ALL);
        return true;
    }

    void CPaintManagerUI::ReapObjects(CControlUI *pControl) {
        if( pControl == m_pEventKey ) m_pEventKey = NULL;

        if( pControl == m_pEventHover ) m_pEventHover = NULL;

        if( pControl == m_pEventClick ) m_pEventClick = NULL;

        if( pControl == m_pFocus ) m_pFocus = NULL;

        KillTimer(pControl);
        const CDuiString &sName = pControl->GetName();

        if( !sName.IsEmpty() ) {
            if( pControl == FindControl(sName) ) m_mNameHash.Remove(sName);
        }

        for( int i = 0; i < m_aAsyncNotify.GetSize(); i++ ) {
            TNotifyUI *pMsg = static_cast<TNotifyUI *>(m_aAsyncNotify[i]);

            if( pMsg->pSender == pControl )
                pMsg->pSender = NULL;
        }
    }

    bool CPaintManagerUI::AddOptionGroup(LPCTSTR pStrGroupName, CControlUI *pControl) {
        LPVOID lp = m_mOptionGroup.Find(pStrGroupName);

        if( lp ) {
            CStdPtrArray *aOptionGroup = static_cast<CStdPtrArray *>(lp);

            for( int i = 0; i < aOptionGroup->GetSize(); i++ ) {
                if( static_cast<CControlUI *>(aOptionGroup->GetAt(i)) == pControl ) {
                    return false;
                }
            }

            aOptionGroup->Add(pControl);
        } else {
            CStdPtrArray *aOptionGroup = new CStdPtrArray(6);
            aOptionGroup->Add(pControl);
            m_mOptionGroup.Insert(pStrGroupName, aOptionGroup);
        }

        return true;
    }

    CStdPtrArray *CPaintManagerUI::GetOptionGroup(LPCTSTR pStrGroupName) {
        LPVOID lp = m_mOptionGroup.Find(pStrGroupName);

        if( lp )
            return static_cast<CStdPtrArray *>(lp);

        return NULL;
    }

    void CPaintManagerUI::RemoveOptionGroup(LPCTSTR pStrGroupName, CControlUI *pControl) {
        LPVOID lp = m_mOptionGroup.Find(pStrGroupName);

        if( lp ) {
            CStdPtrArray *aOptionGroup = static_cast<CStdPtrArray *>(lp);

            if( aOptionGroup == NULL )
                return;

            for( int i = 0; i < aOptionGroup->GetSize(); i++ ) {
                if( static_cast<CControlUI *>(aOptionGroup->GetAt(i)) == pControl ) {
                    aOptionGroup->Remove(i);
                    break;
                }
            }

            if( aOptionGroup->IsEmpty() ) {
                delete aOptionGroup;
                m_mOptionGroup.Remove(pStrGroupName);
            }
        }
    }

    void CPaintManagerUI::RemoveAllOptionGroups() {
        CStdPtrArray *aOptionGroup = NULL;

        for (std::map<CDuiString, LPVOID>::iterator it = m_mOptionGroup.Begin(); it != m_mOptionGroup.End(); it++) {
            aOptionGroup = static_cast<CStdPtrArray *>(it->second);
            delete aOptionGroup;
            it->second = NULL;
        }

        m_mOptionGroup.RemoveAll();
    }

    void CPaintManagerUI::MessageLoop() {
        m_dwUIThreadId = GetCurrentThreadId();
        MSG msg = { 0 };
        while( ::GetMessage(&msg, NULL, 0, 0) ) {
            {
                std::unique_lock<std::mutex> lock(CPaintManagerUI::UIWorksMutex);
                while (!CPaintManagerUI::UIWorks.empty()) {
                    std::function<void()> task = std::move(CPaintManagerUI::UIWorks.front());
                    CPaintManagerUI::UIWorks.pop();
                    task();
                }
            }

            if (msg.message == WM_USER + 1234 && msg.wParam == WM_USER + 1234 &&
                msg.lParam == WM_USER + 1234)
              continue;

            if (!CPaintManagerUI::TranslateMessage(&msg)) {
                ::TranslateMessage(&msg);

                try {
                    ::DispatchMessage(&msg);
                } catch (...) {
#ifdef _DEBUG
                    throw "CPaintManagerUI::MessageLoop";
#endif
                }
            }
        }
    }

    void CPaintManagerUI::Term() {
        // 销毁资源管理器
        CResourceManager::Release();
        CControlFactory::Release();
        //CMenuWnd::DestroyMenu();

        // 清理共享资源
        // 图片
        TImageInfo *data = NULL;

        for (std::map<CDuiString, LPVOID>::iterator it = m_SharedResInfo.m_ImageHash.Begin(); it != m_SharedResInfo.m_ImageHash.End(); it++) {
            data = static_cast<TImageInfo *>(it->second);

            if (data) {
                CRenderEngine::FreeImage(data);
                data = NULL;
            }
        }

        m_SharedResInfo.m_ImageHash.RemoveAll();


        // 字体
        TFontInfo *pFontInfo = NULL;
        for (std::map<CDuiString, LPVOID>::iterator it = m_SharedResInfo.m_CustomFonts.Begin(); it != m_SharedResInfo.m_CustomFonts.End(); it++) {
            pFontInfo = static_cast<TFontInfo *>(it->second);

            if (pFontInfo) {
                ::DeleteObject(pFontInfo->hFont);
                delete pFontInfo;
                pFontInfo = NULL;
            }
        }
        m_SharedResInfo.m_CustomFonts.RemoveAll();

        // 默认字体
        if(m_SharedResInfo.m_DefaultFontInfo.hFont != NULL) {
            ::DeleteObject(m_SharedResInfo.m_DefaultFontInfo.hFont);
        }

        // 样式
        CDuiString *pStyle = NULL;
        for (std::map<CDuiString, LPVOID>::iterator it = m_SharedResInfo.m_StyleHash.Begin(); it != m_SharedResInfo.m_StyleHash.End(); it++) {
            pStyle = static_cast<CDuiString *>(it->second);

            if (pStyle) {
                delete pStyle;
                pStyle = NULL;
            }
        }
        m_SharedResInfo.m_StyleHash.RemoveAll();

        // 样式
        CDuiString *pAttr = NULL;
        for (std::map<CDuiString, LPVOID>::iterator it = m_SharedResInfo.m_AttrHash.Begin(); it != m_SharedResInfo.m_AttrHash.End(); it++) {
            pAttr = static_cast<CDuiString *>(it->second);

            if (pAttr) {
                delete pAttr;
                pAttr = NULL;
            }
        }
        m_SharedResInfo.m_AttrHash.RemoveAll();

        // 关闭ZIP
        if( m_bCachedResourceZip && m_hResourceZip != NULL ) {
            CloseZip((HZIP)m_hResourceZip);
            m_hResourceZip = NULL;
        }
    }

    CDPI *DuiLib::CPaintManagerUI::GetDPIObj() {
        if (!m_pDPI) {
            if (!m_pDPI) {
                m_pDPI = new CDPI();
            }
        }

        return m_pDPI;
    }

    void DuiLib::CPaintManagerUI::SetDPI(int iDPI) {
        int scale1 = GetDPIObj()->GetScale();
        GetDPIObj()->SetScale(iDPI);
        int scale2 = GetDPIObj()->GetScale();
        ResetDPIAssets();
        RECT rcWnd = {0};
        ::GetWindowRect(GetPaintWindow(), &rcWnd);
        RECT  *prcNewWindow = &rcWnd;

        if (!::IsZoomed(GetPaintWindow())) {
            RECT rc = rcWnd;
            rc.right = rcWnd.left + (rcWnd.right - rcWnd.left) * scale2 / scale1;
            rc.bottom = rcWnd.top + (rcWnd.bottom - rcWnd.top) * scale2 / scale1;
            prcNewWindow = &rc;
        }

        SetWindowPos(GetPaintWindow(), NULL, prcNewWindow->left, prcNewWindow->top, prcNewWindow->right - prcNewWindow->left, prcNewWindow->bottom - prcNewWindow->top, SWP_NOZORDER | SWP_NOACTIVATE);

        if (GetRoot() != NULL) GetRoot()->NeedUpdate();

        ::PostMessage(GetPaintWindow(), WM_USER_SET_DPI, 0, 0);
    }

    void DuiLib::CPaintManagerUI::SetAllDPI(int iDPI) {
        for (int i = 0; i < m_aPreMessages.GetSize(); i++) {
            CPaintManagerUI *pManager = static_cast<CPaintManagerUI *>(m_aPreMessages[i]);
            pManager->SetDPI(iDPI);
        }
    }

    void DuiLib::CPaintManagerUI::ResetDPIAssets() {
        RemoveAllDrawInfos();
        RemoveAllImages();

        for (std::map<CDuiString, LPVOID>::iterator it = m_ResInfo.m_CustomFonts.Begin(); it != m_ResInfo.m_CustomFonts.End(); it++) {
            TFontInfo *pFontInfo = static_cast<TFontInfo *>(it->second);
            RebuildFont(pFontInfo);
        }

        RebuildFont(&m_ResInfo.m_DefaultFontInfo);


        for (std::map<CDuiString, LPVOID>::iterator it = m_SharedResInfo.m_CustomFonts.Begin(); it != m_SharedResInfo.m_CustomFonts.End(); it++) {
            TFontInfo *pFontInfo = static_cast<TFontInfo *>(it->second);
            RebuildFont(pFontInfo);
        }

        RebuildFont(&m_SharedResInfo.m_DefaultFontInfo);

        CStdPtrArray *richEditList = FindSubControlsByClass(GetRoot(), _T("RichEditUI"));

        for (int i = 0; i < richEditList->GetSize(); i++) {
            CRichEditUI *pT = static_cast<CRichEditUI *>((*richEditList)[i]);
            pT->SetFont(pT->GetFont());

        }
    }

    void DuiLib::CPaintManagerUI::RebuildFont(TFontInfo *pFontInfo) {
        ::DeleteObject(pFontInfo->hFont);
        LOGFONT lf = { 0 };
        ::GetObject(::GetStockObject(DEFAULT_GUI_FONT), sizeof(LOGFONT), &lf);
        _tcsncpy(lf.lfFaceName, pFontInfo->sFontName, LF_FACESIZE);
        lf.lfCharSet = DEFAULT_CHARSET;
        lf.lfHeight = -GetDPIObj()->Scale(pFontInfo->iSize);
        lf.lfQuality = CLEARTYPE_QUALITY;

        if (pFontInfo->bBold) lf.lfWeight += FW_BOLD;

        if (pFontInfo->bUnderline) lf.lfUnderline = TRUE;

        if (pFontInfo->bItalic) lf.lfItalic = TRUE;

        HFONT hFont = ::CreateFontIndirect(&lf);
        pFontInfo->hFont = hFont;
        ::ZeroMemory(&(pFontInfo->tm), sizeof(pFontInfo->tm));

        if (m_hDcPaint) {
            HFONT hOldFont = (HFONT) ::SelectObject(m_hDcPaint, hFont);
            ::GetTextMetrics(m_hDcPaint, &pFontInfo->tm);
            ::SelectObject(m_hDcPaint, hOldFont);
        }
    }

    CControlUI *CPaintManagerUI::GetFocus() const {
        return m_pFocus;
    }

    void CPaintManagerUI::SetFocus(CControlUI *pControl) {
        // Paint manager window has focus?
        HWND hFocusWnd = ::GetFocus();

        if( hFocusWnd != m_hWndPaint && pControl != m_pFocus )
            ::SetFocus(m_hWndPaint);

        // Already has focus?
        if( pControl == m_pFocus )
            return;

        // Remove focus from old control
        if( m_pFocus != NULL ) {
            TEventUI event = { 0 };
            event.Type = UIEVENT_KILLFOCUS;
            event.pSender = pControl;
            event.dwTimestamp = ::GetTickCount();
            m_pFocus->Event(event);
            SendNotify(m_pFocus, DUI_MSGTYPE_KILLFOCUS);
            m_pFocus = NULL;
        }

        if( pControl == NULL )
            return;

        // Set focus to new control
        if( pControl != NULL
                && pControl->GetManager() == this
                && pControl->IsVisible()
                && pControl->IsEnabled() ) {
            m_pFocus = pControl;
            TEventUI event = { 0 };
            event.Type = UIEVENT_SETFOCUS;
            event.pSender = pControl;
            event.dwTimestamp = ::GetTickCount();
            m_pFocus->Event(event);
            SendNotify(m_pFocus, DUI_MSGTYPE_SETFOCUS);
        }
    }

    void CPaintManagerUI::SetFocusNeeded(CControlUI *pControl) {
        ::SetFocus(m_hWndPaint);

        if( pControl == NULL ) return;

        if( m_pFocus != NULL ) {
            TEventUI event = { 0 };
            event.Type = UIEVENT_KILLFOCUS;
            event.pSender = pControl;
            event.dwTimestamp = ::GetTickCount();
            m_pFocus->Event(event);
            SendNotify(m_pFocus, DUI_MSGTYPE_KILLFOCUS);
            m_pFocus = NULL;
        }

        FINDTABINFO info = { 0 };
        info.pFocus = pControl;
        info.bForward = false;
        m_pFocus = m_pRoot->FindControl(__FindControlFromTab, &info, UIFIND_VISIBLE | UIFIND_ENABLED | UIFIND_ME_FIRST);
        m_bFocusNeeded = true;

        if( m_pRoot != NULL ) m_pRoot->NeedUpdate();
    }

    bool CPaintManagerUI::SetTimer(CControlUI *pControl, UINT nTimerID, UINT uElapse) {
        ASSERT(pControl != NULL);
        ASSERT(uElapse > 0);

        for( int i = 0; i < m_aTimers.GetSize(); i++ ) {
            TIMERINFO *pTimer = static_cast<TIMERINFO *>(m_aTimers[i]);

            if( pTimer->pSender == pControl
                    && pTimer->hWnd == m_hWndPaint
                    && pTimer->nLocalID == nTimerID ) {
                if( pTimer->bKilled == true ) {
                    if( ::SetTimer(m_hWndPaint, pTimer->uWinTimer, uElapse, NULL) ) {
                        pTimer->bKilled = false;
                        return true;
                    }

                    return false;
                }

                return false;
            }
        }

        m_uTimerID = (++m_uTimerID) % 0xF0; //0xf1-0xfe特殊用途

        if( !::SetTimer(m_hWndPaint, m_uTimerID, uElapse, NULL) ) return FALSE;

        TIMERINFO *pTimer = new TIMERINFO;

        if( pTimer == NULL ) return FALSE;

        pTimer->hWnd = m_hWndPaint;
        pTimer->pSender = pControl;
        pTimer->nLocalID = nTimerID;
        pTimer->uWinTimer = m_uTimerID;
        pTimer->bKilled = false;
        return m_aTimers.Add(pTimer);
    }

    bool CPaintManagerUI::KillTimer(CControlUI *pControl, UINT nTimerID) {
        ASSERT(pControl != NULL);

        for( int i = 0; i < m_aTimers.GetSize(); i++ ) {
            TIMERINFO *pTimer = static_cast<TIMERINFO *>(m_aTimers[i]);

            if( pTimer->pSender == pControl
                    && pTimer->hWnd == m_hWndPaint
                    && pTimer->nLocalID == nTimerID ) {
                if( pTimer->bKilled == false ) {
                    if( ::IsWindow(m_hWndPaint) ) ::KillTimer(pTimer->hWnd, pTimer->uWinTimer);

                    pTimer->bKilled = true;
                    return true;
                }
            }
        }

        return false;
    }

    void CPaintManagerUI::KillTimer(CControlUI *pControl) {
        ASSERT(pControl != NULL);
        int count = m_aTimers.GetSize();

        for( int i = 0, j = 0; i < count; i++ ) {
            TIMERINFO *pTimer = static_cast<TIMERINFO *>(m_aTimers[i - j]);

            if( pTimer->pSender == pControl && pTimer->hWnd == m_hWndPaint ) {
                if( pTimer->bKilled == false ) ::KillTimer(pTimer->hWnd, pTimer->uWinTimer);

                delete pTimer;
                m_aTimers.Remove(i - j);
                j++;
            }
        }
    }

    void CPaintManagerUI::RemoveAllTimers() {
        for( int i = 0; i < m_aTimers.GetSize(); i++ ) {
            TIMERINFO *pTimer = static_cast<TIMERINFO *>(m_aTimers[i]);

            if( pTimer->hWnd == m_hWndPaint ) {
                if( pTimer->bKilled == false ) {
                    if( ::IsWindow(m_hWndPaint) ) ::KillTimer(m_hWndPaint, pTimer->uWinTimer);
                }

                delete pTimer;
            }
        }

        m_aTimers.Empty();
    }

    void CPaintManagerUI::SetCapture() {
        ::SetCapture(m_hWndPaint);
        m_bMouseCapture = true;
    }

    void CPaintManagerUI::ReleaseCapture() {
        ::ReleaseCapture();
        m_bMouseCapture = false;
        m_bDragMode = false;
    }

    bool CPaintManagerUI::IsCaptured() {
        return m_bMouseCapture;
    }

    bool CPaintManagerUI::IsPainting() {
        return m_bIsPainting;
    }

    void CPaintManagerUI::SetPainting(bool bIsPainting) {
        m_bIsPainting = bIsPainting;
    }

    bool CPaintManagerUI::SetNextTabControl(bool bForward) {
        // If we're in the process of restructuring the layout we can delay the
        // focus calulation until the next repaint.
        if( m_bUpdateNeeded && bForward ) {
            m_bFocusNeeded = true;
            ::InvalidateRect(m_hWndPaint, NULL, FALSE);
            return true;
        }

        // Find next/previous tabbable control
        FINDTABINFO info1 = { 0 };
        info1.pFocus = m_pFocus;
        info1.bForward = bForward;
        CControlUI *pControl = m_pRoot->FindControl(__FindControlFromTab, &info1, UIFIND_VISIBLE | UIFIND_ENABLED | UIFIND_ME_FIRST);

        if( pControl == NULL ) {
            if( bForward ) {
                // Wrap around
                FINDTABINFO info2 = { 0 };
                info2.pFocus = bForward ? NULL : info1.pLast;
                info2.bForward = bForward;
                pControl = m_pRoot->FindControl(__FindControlFromTab, &info2, UIFIND_VISIBLE | UIFIND_ENABLED | UIFIND_ME_FIRST);
            } else {
                pControl = info1.pLast;
            }
        }

        if( pControl != NULL ) SetFocus(pControl);

        m_bFocusNeeded = false;
        return true;
    }

    bool CPaintManagerUI::AddNotifier(INotifyUI *pNotifier) {
        ASSERT(m_aNotifiers.Find(pNotifier) < 0);
        return m_aNotifiers.Add(pNotifier);
    }

    bool CPaintManagerUI::RemoveNotifier(INotifyUI *pNotifier) {
        for( int i = 0; i < m_aNotifiers.GetSize(); i++ ) {
            if( static_cast<INotifyUI *>(m_aNotifiers[i]) == pNotifier ) {
                return m_aNotifiers.Remove(i);
            }
        }

        return false;
    }

    bool CPaintManagerUI::AddPreMessageFilter(IMessageFilterUI *pFilter) {
        ASSERT(m_aPreMessageFilters.Find(pFilter) < 0);
        return m_aPreMessageFilters.Add(pFilter);
    }

    bool CPaintManagerUI::RemovePreMessageFilter(IMessageFilterUI *pFilter) {
        for( int i = 0; i < m_aPreMessageFilters.GetSize(); i++ ) {
            if( static_cast<IMessageFilterUI *>(m_aPreMessageFilters[i]) == pFilter ) {
                return m_aPreMessageFilters.Remove(i);
            }
        }

        return false;
    }

    bool CPaintManagerUI::AddMessageFilter(IMessageFilterUI *pFilter) {
        ASSERT(m_aMessageFilters.Find(pFilter) < 0);
        return m_aMessageFilters.Add(pFilter);
    }

    bool CPaintManagerUI::RemoveMessageFilter(IMessageFilterUI *pFilter) {
        for( int i = 0; i < m_aMessageFilters.GetSize(); i++ ) {
            if( static_cast<IMessageFilterUI *>(m_aMessageFilters[i]) == pFilter ) {
                return m_aMessageFilters.Remove(i);
            }
        }

        return false;
    }

    int CPaintManagerUI::GetPostPaintCount() const {
        return m_aPostPaintControls.GetSize();
    }

    bool CPaintManagerUI::IsPostPaint(CControlUI *pControl) {
        return m_aPostPaintControls.Find(pControl) >= 0;
    }

    bool CPaintManagerUI::AddPostPaint(CControlUI *pControl) {
        ASSERT(m_aPostPaintControls.Find(pControl) < 0);
        return m_aPostPaintControls.Add(pControl);
    }

    bool CPaintManagerUI::RemovePostPaint(CControlUI *pControl) {
        for( int i = 0; i < m_aPostPaintControls.GetSize(); i++ ) {
            if( static_cast<CControlUI *>(m_aPostPaintControls[i]) == pControl ) {
                return m_aPostPaintControls.Remove(i);
            }
        }

        return false;
    }

    bool CPaintManagerUI::SetPostPaintIndex(CControlUI *pControl, int iIndex) {
        RemovePostPaint(pControl);
        return m_aPostPaintControls.InsertAt(iIndex, pControl);
    }

    int CPaintManagerUI::GetNativeWindowCount() const {
        return m_aNativeWindow.GetSize();
    }

    bool CPaintManagerUI::AddNativeWindow(CControlUI *pControl, HWND hChildWnd) {
        if (pControl == NULL || hChildWnd == NULL) return false;

        RECT rcChildWnd = GetNativeWindowRect(hChildWnd);
        Invalidate(rcChildWnd);

        if (m_aNativeWindow.Find(hChildWnd) >= 0) return false;

        if (m_aNativeWindow.Add(hChildWnd)) {
            m_aNativeWindowControl.Add(pControl);
            return true;
        }

        return false;
    }

    bool CPaintManagerUI::RemoveNativeWindow(HWND hChildWnd) {
        for( int i = 0; i < m_aNativeWindow.GetSize(); i++ ) {
            if( static_cast<HWND>(m_aNativeWindow[i]) == hChildWnd ) {
                if( m_aNativeWindow.Remove(i) ) {
                    m_aNativeWindowControl.Remove(i);
                    return true;
                }

                return false;
            }
        }

        return false;
    }

    RECT CPaintManagerUI::GetNativeWindowRect(HWND hChildWnd) {
        RECT rcChildWnd;
        ::GetWindowRect(hChildWnd, &rcChildWnd);
        ::ScreenToClient(m_hWndPaint, (LPPOINT)(&rcChildWnd));
        ::ScreenToClient(m_hWndPaint, (LPPOINT)(&rcChildWnd) + 1);
        return rcChildWnd;
    }

    void CPaintManagerUI::AddDelayedCleanup(CControlUI *pControl) {
        if (pControl == NULL) return;

        pControl->SetManager(this, NULL, false);
        m_aDelayedCleanup.Add(pControl);
        PostAsyncNotify();
    }

    void CPaintManagerUI::AddMouseLeaveNeeded(CControlUI *pControl) {
        if (pControl == NULL) return;

        for( int i = 0; i < m_aNeedMouseLeaveNeeded.GetSize(); i++ ) {
            if( static_cast<CControlUI *>(m_aNeedMouseLeaveNeeded[i]) == pControl ) {
                return;
            }
        }

        m_aNeedMouseLeaveNeeded.Add(pControl);
    }

    bool CPaintManagerUI::RemoveMouseLeaveNeeded(CControlUI *pControl) {
        if (pControl == NULL) return false;

        for( int i = 0; i < m_aNeedMouseLeaveNeeded.GetSize(); i++ ) {
            if( static_cast<CControlUI *>(m_aNeedMouseLeaveNeeded[i]) == pControl ) {
                return m_aNeedMouseLeaveNeeded.Remove(i);
            }
        }

        return false;
    }

    void CPaintManagerUI::SendNotify(CControlUI *pControl, LPCTSTR pstrMessage, WPARAM wParam /*= 0*/, LPARAM lParam /*= 0*/, bool bAsync /*= false*/) {
        TNotifyUI Msg;
        Msg.pSender = pControl;
        Msg.sType = pstrMessage;
        Msg.wParam = wParam;
        Msg.lParam = lParam;
        SendNotify(Msg, bAsync);
    }

    void CPaintManagerUI::SendNotify(TNotifyUI &Msg, bool bAsync /*= false*/) {
        Msg.ptMouse = m_ptLastMousePos;
        Msg.dwTimestamp = ::GetTickCount();

        if( m_bUsedVirtualWnd ) {
            Msg.sVirtualWnd = Msg.pSender->GetVirtualWnd();
        }

        if( !bAsync ) {
            // Send to all listeners
            if( Msg.pSender != NULL ) {
                if( Msg.pSender->m_OnNotifyDelegate ) Msg.pSender->m_OnNotifyDelegate(&Msg);
            }

            for( int i = 0; i < m_aNotifiers.GetSize(); i++ ) {
                static_cast<INotifyUI *>(m_aNotifiers[i])->Notify(Msg);
            }
        } else {
            TNotifyUI *pMsg = new TNotifyUI;
            pMsg->pSender = Msg.pSender;
            pMsg->sType = Msg.sType;
            pMsg->wParam = Msg.wParam;
            pMsg->lParam = Msg.lParam;
            pMsg->ptMouse = Msg.ptMouse;
            pMsg->dwTimestamp = Msg.dwTimestamp;
            m_aAsyncNotify.Add(pMsg);

            PostAsyncNotify();
        }
    }

    bool CPaintManagerUI::IsForceUseSharedRes() const {
        return m_bForceUseSharedRes;
    }

    void CPaintManagerUI::SetForceUseSharedRes(bool bForce) {
        m_bForceUseSharedRes = bForce;
    }

    DWORD CPaintManagerUI::GetDefaultDisabledColor() const {
        return m_ResInfo.m_dwDefaultDisabledColor;
    }

    void CPaintManagerUI::SetDefaultDisabledColor(DWORD dwColor, bool bShared) {
        if (bShared) {
            if (m_ResInfo.m_dwDefaultDisabledColor == m_SharedResInfo.m_dwDefaultDisabledColor)
                m_ResInfo.m_dwDefaultDisabledColor = dwColor;

            m_SharedResInfo.m_dwDefaultDisabledColor = dwColor;
        } else {
            m_ResInfo.m_dwDefaultDisabledColor = dwColor;
        }
    }

    DWORD CPaintManagerUI::GetDefaultFontColor() const {
        return m_ResInfo.m_dwDefaultFontColor;
    }

    void CPaintManagerUI::SetDefaultFontColor(DWORD dwColor, bool bShared) {
        if (bShared) {
            if (m_ResInfo.m_dwDefaultFontColor == m_SharedResInfo.m_dwDefaultFontColor)
                m_ResInfo.m_dwDefaultFontColor = dwColor;

            m_SharedResInfo.m_dwDefaultFontColor = dwColor;
        } else {
            m_ResInfo.m_dwDefaultFontColor = dwColor;
        }
    }

    DWORD CPaintManagerUI::GetDefaultLinkFontColor() const {
        return m_ResInfo.m_dwDefaultLinkFontColor;
    }

    void CPaintManagerUI::SetDefaultLinkFontColor(DWORD dwColor, bool bShared) {
        if (bShared) {
            if (m_ResInfo.m_dwDefaultLinkFontColor == m_SharedResInfo.m_dwDefaultLinkFontColor)
                m_ResInfo.m_dwDefaultLinkFontColor = dwColor;

            m_SharedResInfo.m_dwDefaultLinkFontColor = dwColor;
        } else {
            m_ResInfo.m_dwDefaultLinkFontColor = dwColor;
        }
    }

    DWORD CPaintManagerUI::GetDefaultLinkHoverFontColor() const {
        return m_ResInfo.m_dwDefaultLinkHoverFontColor;
    }

    void CPaintManagerUI::SetDefaultLinkHoverFontColor(DWORD dwColor, bool bShared) {
        if (bShared) {
            if (m_ResInfo.m_dwDefaultLinkHoverFontColor == m_SharedResInfo.m_dwDefaultLinkHoverFontColor)
                m_ResInfo.m_dwDefaultLinkHoverFontColor = dwColor;

            m_SharedResInfo.m_dwDefaultLinkHoverFontColor = dwColor;
        } else {
            m_ResInfo.m_dwDefaultLinkHoverFontColor = dwColor;
        }
    }

    DWORD CPaintManagerUI::GetDefaultSelectedBkColor() const {
        return m_ResInfo.m_dwDefaultSelectedBkColor;
    }

    void CPaintManagerUI::SetDefaultSelectedBkColor(DWORD dwColor, bool bShared) {
        if (bShared) {
            if (m_ResInfo.m_dwDefaultSelectedBkColor == m_SharedResInfo.m_dwDefaultSelectedBkColor)
                m_ResInfo.m_dwDefaultSelectedBkColor = dwColor;

            m_SharedResInfo.m_dwDefaultSelectedBkColor = dwColor;
        } else {
            m_ResInfo.m_dwDefaultSelectedBkColor = dwColor;
        }
    }

    TFontInfo *CPaintManagerUI::GetDefaultFontInfo() {
        if (m_ResInfo.m_DefaultFontInfo.sFontName.IsEmpty()) {
            if( m_SharedResInfo.m_DefaultFontInfo.tm.tmHeight == 0 ) {
                HFONT hOldFont = (HFONT) ::SelectObject(m_hDcPaint, m_SharedResInfo.m_DefaultFontInfo.hFont);
                ::GetTextMetrics(m_hDcPaint, &m_SharedResInfo.m_DefaultFontInfo.tm);
                ::SelectObject(m_hDcPaint, hOldFont);
            }

            return &m_SharedResInfo.m_DefaultFontInfo;
        } else {
            if( m_ResInfo.m_DefaultFontInfo.tm.tmHeight == 0 ) {
                HFONT hOldFont = (HFONT) ::SelectObject(m_hDcPaint, m_ResInfo.m_DefaultFontInfo.hFont);
                ::GetTextMetrics(m_hDcPaint, &m_ResInfo.m_DefaultFontInfo.tm);
                ::SelectObject(m_hDcPaint, hOldFont);
            }

            return &m_ResInfo.m_DefaultFontInfo;
        }
    }

    void CPaintManagerUI::SetDefaultFont(LPCTSTR pStrFontName, int nSize, bool bBold, bool bUnderline, bool bItalic, bool bShared) {
        LOGFONT lf = { 0 };
        ::GetObject(::GetStockObject(DEFAULT_GUI_FONT), sizeof(LOGFONT), &lf);

        if(lstrlen(pStrFontName) > 0) {
            TCHAR szFaceName[32] = {0};//_T("@");
            _tcsncat(szFaceName, pStrFontName, LF_FACESIZE);
            _tcsncpy(lf.lfFaceName, szFaceName, LF_FACESIZE);
        }

        lf.lfCharSet = DEFAULT_CHARSET;
        lf.lfHeight = -GetDPIObj()->Scale(nSize);;

        if( bBold ) lf.lfWeight += FW_BOLD;

        if( bUnderline ) lf.lfUnderline = TRUE;

        if( bItalic ) lf.lfItalic = TRUE;

        HFONT hFont = ::CreateFontIndirect(&lf);

        if( hFont == NULL ) return;

        if (bShared) {
            ::DeleteObject(m_SharedResInfo.m_DefaultFontInfo.hFont);
            m_SharedResInfo.m_DefaultFontInfo.hFont = hFont;
            m_SharedResInfo.m_DefaultFontInfo.sFontName = lf.lfFaceName;
            m_SharedResInfo.m_DefaultFontInfo.iSize = nSize;
            m_SharedResInfo.m_DefaultFontInfo.bBold = bBold;
            m_SharedResInfo.m_DefaultFontInfo.bUnderline = bUnderline;
            m_SharedResInfo.m_DefaultFontInfo.bItalic = bItalic;
            ::ZeroMemory(&m_SharedResInfo.m_DefaultFontInfo.tm, sizeof(m_SharedResInfo.m_DefaultFontInfo.tm));

            if( m_hDcPaint ) {
                HFONT hOldFont = (HFONT) ::SelectObject(m_hDcPaint, hFont);
                ::GetTextMetrics(m_hDcPaint, &m_SharedResInfo.m_DefaultFontInfo.tm);
                ::SelectObject(m_hDcPaint, hOldFont);
            }
        } else {
            ::DeleteObject(m_ResInfo.m_DefaultFontInfo.hFont);
            m_ResInfo.m_DefaultFontInfo.hFont = hFont;
            m_ResInfo.m_DefaultFontInfo.sFontName = lf.lfFaceName;
            m_ResInfo.m_DefaultFontInfo.iSize = nSize;
            m_ResInfo.m_DefaultFontInfo.bBold = bBold;
            m_ResInfo.m_DefaultFontInfo.bUnderline = bUnderline;
            m_ResInfo.m_DefaultFontInfo.bItalic = bItalic;
            ::ZeroMemory(&m_ResInfo.m_DefaultFontInfo.tm, sizeof(m_ResInfo.m_DefaultFontInfo.tm));

            if( m_hDcPaint ) {
                HFONT hOldFont = (HFONT) ::SelectObject(m_hDcPaint, hFont);
                ::GetTextMetrics(m_hDcPaint, &m_ResInfo.m_DefaultFontInfo.tm);
                ::SelectObject(m_hDcPaint, hOldFont);
            }
        }
    }

    DWORD CPaintManagerUI::GetCustomFontCount(bool bShared) const {
        if (bShared)
            return m_SharedResInfo.m_CustomFonts.GetSize();
        else
            return m_ResInfo.m_CustomFonts.GetSize();
    }

    HFONT CPaintManagerUI::AddFont(int id, LPCTSTR pStrFontName, int nSize, bool bBold, bool bUnderline, bool bItalic, bool bShared) {
        LOGFONT lf = { 0 };
        ::GetObject(::GetStockObject(DEFAULT_GUI_FONT), sizeof(LOGFONT), &lf);

        if(lstrlen(pStrFontName) > 0) {
            TCHAR szFaceName[32] = {0};//_T("@");
            _tcsncat(szFaceName, pStrFontName, LF_FACESIZE);
            _tcsncpy(lf.lfFaceName, szFaceName, LF_FACESIZE);
        }

        lf.lfCharSet = DEFAULT_CHARSET;
        lf.lfHeight = -GetDPIObj()->Scale(nSize);

        if( bBold ) lf.lfWeight = FW_BOLD;

        if( bUnderline ) lf.lfUnderline = TRUE;

        if( bItalic ) lf.lfItalic = TRUE;

        HFONT hFont = ::CreateFontIndirect(&lf);

        if( hFont == NULL ) return NULL;

        TFontInfo *pFontInfo = new TFontInfo;

        if( !pFontInfo ) return false;

        pFontInfo->hFont = hFont;
        pFontInfo->sFontName = lf.lfFaceName;
        pFontInfo->iSize = nSize;
        pFontInfo->bBold = bBold;
        pFontInfo->bUnderline = bUnderline;
        pFontInfo->bItalic = bItalic;

        if( m_hDcPaint ) {
            HFONT hOldFont = (HFONT) ::SelectObject(m_hDcPaint, hFont);
            ::GetTextMetrics(m_hDcPaint, &pFontInfo->tm);
            ::SelectObject(m_hDcPaint, hOldFont);
        }

        TCHAR idBuffer[16];
        ::ZeroMemory(idBuffer, sizeof(idBuffer));
        _itot(id, idBuffer, 10);

        if (bShared || m_bForceUseSharedRes) {
            TFontInfo *pOldFontInfo = static_cast<TFontInfo *>(m_SharedResInfo.m_CustomFonts.Find(idBuffer));

            if (pOldFontInfo) {
                ::DeleteObject(pOldFontInfo->hFont);
                delete pOldFontInfo;
                m_SharedResInfo.m_CustomFonts.Remove(idBuffer);
            }

            if( !m_SharedResInfo.m_CustomFonts.Insert(idBuffer, pFontInfo) ) {
                ::DeleteObject(hFont);
                delete pFontInfo;
                return NULL;
            }
        } else {
            TFontInfo *pOldFontInfo = static_cast<TFontInfo *>(m_ResInfo.m_CustomFonts.Find(idBuffer));

            if (pOldFontInfo) {
                ::DeleteObject(pOldFontInfo->hFont);
                delete pOldFontInfo;
                m_ResInfo.m_CustomFonts.Remove(idBuffer);
            }

            if( !m_ResInfo.m_CustomFonts.Insert(idBuffer, pFontInfo) ) {
                ::DeleteObject(hFont);
                delete pFontInfo;
                return NULL;
            }
        }

        return hFont;
    }

    void CPaintManagerUI::AddFontArray(LPCTSTR pstrPath) {
        LPBYTE pData = NULL;
        DWORD dwSize = 0;

        do {
            CDuiString sFile = CPaintManagerUI::GetResourcePath();

            if (CPaintManagerUI::GetResourceZip().IsEmpty()) {
                sFile += pstrPath;
                HANDLE hFile = ::CreateFile(sFile, GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, \
                                            FILE_ATTRIBUTE_NORMAL, NULL);

                if (hFile == INVALID_HANDLE_VALUE) break;

                dwSize = ::GetFileSize(hFile, NULL);

                if (dwSize == 0) break;

                DWORD dwRead = 0;
                pData = new BYTE[dwSize];
                ::ReadFile(hFile, pData, dwSize, &dwRead, NULL);
                ::CloseHandle(hFile);

                if (dwRead != dwSize) {
                    delete[] pData;
                    pData = NULL;
                    break;
                }
            } else {
                sFile += CPaintManagerUI::GetResourceZip();
                HZIP hz = NULL;

                if (CPaintManagerUI::IsCachedResourceZip()) hz = (HZIP)CPaintManagerUI::GetResourceZipHandle();
                else {
                    CDuiString sFilePwd = CPaintManagerUI::GetResourceZipPwd();
#ifdef UNICODE
                    std::string pwd = Unicode2Ansi(sFilePwd.GetData());
                    hz = OpenZip(sFile, pwd.c_str());
#else
                    hz = OpenZip(sFile, sFilePwd);
#endif
                }

                if (hz == NULL) break;

                ZIPENTRY ze;
                int i = 0;
                CDuiString key = pstrPath;
                key.Replace(_T("\\"), _T("/"));

                if (FindZipItem(hz, key, true, &i, &ze) != 0) break;

                dwSize = ze.unc_size;

                if (dwSize == 0) break;

                pData = new BYTE[dwSize];
                int res = UnzipItem(hz, i, pData, dwSize);

                if (res != 0x00000000 && res != 0x00000600) {
                    delete[] pData;
                    pData = NULL;

                    if (!CPaintManagerUI::IsCachedResourceZip()) CloseZip(hz);

                    break;
                }

                if (!CPaintManagerUI::IsCachedResourceZip()) CloseZip(hz);
            }

        } while (0);

        while (!pData) {
            //读不到图片, 则直接去读取bitmap.m_lpstr指向的路径
            HANDLE hFile = ::CreateFile(pstrPath, GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);

            if (hFile == INVALID_HANDLE_VALUE) break;

            dwSize = ::GetFileSize(hFile, NULL);

            if (dwSize == 0) break;

            DWORD dwRead = 0;
            pData = new BYTE[dwSize];
            ::ReadFile(hFile, pData, dwSize, &dwRead, NULL);
            ::CloseHandle(hFile);

            if (dwRead != dwSize) {
                delete[] pData;
                pData = NULL;
            }

            break;
        }

        DWORD nFonts;
        HANDLE hFont = ::AddFontMemResourceEx(pData, dwSize, NULL, &nFonts);
        delete[] pData;
        pData = NULL;
        m_aFonts.Add(hFont);
    }

    HFONT CPaintManagerUI::GetFont(int id) {
        if (id < 0) return GetDefaultFontInfo()->hFont;

        TCHAR idBuffer[16];
        ::ZeroMemory(idBuffer, sizeof(idBuffer));
        _itot(id, idBuffer, 10);
        TFontInfo *pFontInfo = static_cast<TFontInfo *>(m_ResInfo.m_CustomFonts.Find(idBuffer));

        if( !pFontInfo ) pFontInfo = static_cast<TFontInfo *>(m_SharedResInfo.m_CustomFonts.Find(idBuffer));

        if (!pFontInfo) return GetDefaultFontInfo()->hFont;

        return pFontInfo->hFont;
    }

    HFONT CPaintManagerUI::GetFont(LPCTSTR pStrFontName, int nSize, bool bBold, bool bUnderline, bool bItalic) {
        TFontInfo *pFontInfo = NULL;

        for (std::map<CDuiString, LPVOID>::iterator it = m_ResInfo.m_CustomFonts.Begin(); it != m_ResInfo.m_CustomFonts.End(); it++) {
            pFontInfo = static_cast<TFontInfo *>(it->second);

            if (pFontInfo && pFontInfo->sFontName == pStrFontName && pFontInfo->iSize == nSize &&
                    pFontInfo->bBold == bBold && pFontInfo->bUnderline == bUnderline && pFontInfo->bItalic == bItalic)
                return pFontInfo->hFont;
        }


        for (std::map<CDuiString, LPVOID>::iterator it = m_SharedResInfo.m_CustomFonts.Begin(); it != m_SharedResInfo.m_CustomFonts.End(); it++) {
            pFontInfo = static_cast<TFontInfo *>(it->second);

            if (pFontInfo && pFontInfo->sFontName == pStrFontName && pFontInfo->iSize == nSize &&
                    pFontInfo->bBold == bBold && pFontInfo->bUnderline == bUnderline && pFontInfo->bItalic == bItalic)
                return pFontInfo->hFont;
        }

        return NULL;
    }

    int CPaintManagerUI::GetFontIndex(HFONT hFont, bool bShared) {
        TFontInfo *pFontInfo = NULL;

        if (bShared) {
            for (std::map<CDuiString, LPVOID>::iterator it = m_SharedResInfo.m_CustomFonts.Begin(); it != m_SharedResInfo.m_CustomFonts.End(); it++) {
                pFontInfo = static_cast<TFontInfo *>(it->second);

                if (pFontInfo && pFontInfo->hFont == hFont)
                    return _ttoi(it->first);
            }
        } else {
            for (std::map<CDuiString, LPVOID>::iterator it = m_ResInfo.m_CustomFonts.Begin(); it != m_ResInfo.m_CustomFonts.End(); it++) {
                pFontInfo = static_cast<TFontInfo *>(it->second);

                if (pFontInfo && pFontInfo->hFont == hFont)
                    return _ttoi(it->first);
            }
        }

        return -1;
    }

    int CPaintManagerUI::GetFontIndex(LPCTSTR pStrFontName, int nSize, bool bBold, bool bUnderline, bool bItalic, bool bShared) {
        TFontInfo *pFontInfo = NULL;

        if (bShared) {
            for (std::map<CDuiString, LPVOID>::iterator it = m_SharedResInfo.m_CustomFonts.Begin(); it != m_SharedResInfo.m_CustomFonts.End(); it++) {
                pFontInfo = static_cast<TFontInfo *>(it->second);

                if (pFontInfo && pFontInfo->sFontName == pStrFontName && pFontInfo->iSize == nSize &&
                        pFontInfo->bBold == bBold && pFontInfo->bUnderline == bUnderline && pFontInfo->bItalic == bItalic)
                    return _ttoi(it->first);
            }
        } else {
            for (std::map<CDuiString, LPVOID>::iterator it = m_ResInfo.m_CustomFonts.Begin(); it != m_ResInfo.m_CustomFonts.End(); it++) {
                pFontInfo = static_cast<TFontInfo *>(it->second);

                if (pFontInfo && pFontInfo->sFontName == pStrFontName && pFontInfo->iSize == nSize &&
                        pFontInfo->bBold == bBold && pFontInfo->bUnderline == bUnderline && pFontInfo->bItalic == bItalic)
                    return _ttoi(it->first);
            }
        }

        return -1;
    }

    void CPaintManagerUI::RemoveFont(HFONT hFont, bool bShared) {
        TFontInfo *pFontInfo = NULL;

        if (bShared) {
            for (std::map<CDuiString, LPVOID>::iterator it = m_SharedResInfo.m_CustomFonts.Begin(); it != m_SharedResInfo.m_CustomFonts.End(); it++) {
                pFontInfo = static_cast<TFontInfo *>(it->second);

                if (pFontInfo && pFontInfo->hFont == hFont) {
                    ::DeleteObject(pFontInfo->hFont);
                    delete pFontInfo;
                    m_SharedResInfo.m_CustomFonts.Remove(it->first);
                    return;
                }
            }
        } else {
            for (std::map<CDuiString, LPVOID>::iterator it = m_ResInfo.m_CustomFonts.Begin(); it != m_ResInfo.m_CustomFonts.End(); it++) {
                pFontInfo = static_cast<TFontInfo *>(it->second);

                if (pFontInfo && pFontInfo->hFont == hFont) {
                    ::DeleteObject(pFontInfo->hFont);
                    delete pFontInfo;
                    m_ResInfo.m_CustomFonts.Remove(it->first);
                    return;
                }
            }
        }
    }

    void CPaintManagerUI::RemoveFont(int id, bool bShared) {
        TCHAR idBuffer[16];
        ::ZeroMemory(idBuffer, sizeof(idBuffer));
        _itot(id, idBuffer, 10);

        TFontInfo *pFontInfo = NULL;

        if (bShared) {
            pFontInfo = static_cast<TFontInfo *>(m_SharedResInfo.m_CustomFonts.Find(idBuffer));

            if (pFontInfo) {
                ::DeleteObject(pFontInfo->hFont);
                delete pFontInfo;
                m_SharedResInfo.m_CustomFonts.Remove(idBuffer);
            }
        } else {
            pFontInfo = static_cast<TFontInfo *>(m_ResInfo.m_CustomFonts.Find(idBuffer));

            if (pFontInfo) {
                ::DeleteObject(pFontInfo->hFont);
                delete pFontInfo;
                m_ResInfo.m_CustomFonts.Remove(idBuffer);
            }
        }
    }

    void CPaintManagerUI::RemoveAllFonts(bool bShared) {
        TFontInfo *pFontInfo;

        if (bShared) {
            for (std::map<CDuiString, LPVOID>::iterator it = m_SharedResInfo.m_CustomFonts.Begin(); it != m_SharedResInfo.m_CustomFonts.End(); it++) {
                pFontInfo = static_cast<TFontInfo *>(it->second);

                if (pFontInfo) {
                    ::DeleteObject(pFontInfo->hFont);
                    delete pFontInfo;
                }
            }
            m_SharedResInfo.m_CustomFonts.RemoveAll();
        } else {
            for (std::map<CDuiString, LPVOID>::iterator it = m_ResInfo.m_CustomFonts.Begin(); it != m_ResInfo.m_CustomFonts.End(); it++) {
                pFontInfo = static_cast<TFontInfo *>(it->second);

                if (pFontInfo) {
                    ::DeleteObject(pFontInfo->hFont);
                    delete pFontInfo;
                }
            }
            m_ResInfo.m_CustomFonts.RemoveAll();
        }
    }

    TFontInfo *CPaintManagerUI::GetFontInfo(int id) {
        if (id < 0) return GetDefaultFontInfo();

        TCHAR idBuffer[16];
        ::ZeroMemory(idBuffer, sizeof(idBuffer));
        _itot(id, idBuffer, 10);
        TFontInfo *pFontInfo = static_cast<TFontInfo *>(m_ResInfo.m_CustomFonts.Find(idBuffer));

        if (!pFontInfo) pFontInfo = static_cast<TFontInfo *>(m_SharedResInfo.m_CustomFonts.Find(idBuffer));

        if (!pFontInfo) pFontInfo = GetDefaultFontInfo();

        if (pFontInfo->tm.tmHeight == 0) {
            HFONT hOldFont = (HFONT) ::SelectObject(m_hDcPaint, pFontInfo->hFont);
            ::GetTextMetrics(m_hDcPaint, &pFontInfo->tm);
            ::SelectObject(m_hDcPaint, hOldFont);
        }

        return pFontInfo;
    }

    TFontInfo *CPaintManagerUI::GetFontInfo(HFONT hFont) {
        TFontInfo *pFontInfo = NULL;

        for (std::map<CDuiString, LPVOID>::iterator it = m_ResInfo.m_CustomFonts.Begin(); it != m_ResInfo.m_CustomFonts.End(); it++) {
            TFontInfo *tmp = static_cast<TFontInfo *>(it->second);

            if (tmp && tmp->hFont == hFont) {
                pFontInfo = tmp;
                break;
            }
        }

        if (!pFontInfo) {
            for (std::map<CDuiString, LPVOID>::iterator it = m_SharedResInfo.m_CustomFonts.Begin(); it != m_SharedResInfo.m_CustomFonts.End(); it++) {
                TFontInfo *tmp = static_cast<TFontInfo *>(it->second);

                if (tmp && tmp->hFont == hFont) {
                    pFontInfo = tmp;
                    break;
                }
            }
        }

        if (!pFontInfo)
            pFontInfo = GetDefaultFontInfo();

        if (pFontInfo) {
            if (pFontInfo->tm.tmHeight == 0) {
                HFONT hOldFont = (HFONT) ::SelectObject(m_hDcPaint, pFontInfo->hFont);
                ::GetTextMetrics(m_hDcPaint, &pFontInfo->tm);
                ::SelectObject(m_hDcPaint, hOldFont);
            }
        }

        return pFontInfo;
    }

    const TImageInfo *CPaintManagerUI::GetImage(LPCTSTR bitmap) {
        TImageInfo *data = static_cast<TImageInfo *>(m_ResInfo.m_ImageHash.Find(bitmap));

        if( !data )
            data = static_cast<TImageInfo *>(m_SharedResInfo.m_ImageHash.Find(bitmap));

        return data;
    }

    const TImageInfo *CPaintManagerUI::GetImageEx(LPCTSTR bitmap, LPCTSTR type, DWORD mask, bool bUseHSL, HINSTANCE instance) {
        const TImageInfo *data = GetImage(bitmap);

        if( !data ) {
            if( AddImage(bitmap, type, mask, bUseHSL, false, instance) ) {
                if (m_bForceUseSharedRes)
                    data = static_cast<TImageInfo *>(m_SharedResInfo.m_ImageHash.Find(bitmap));
                else
                    data = static_cast<TImageInfo *>(m_ResInfo.m_ImageHash.Find(bitmap));
            }
        }

        return data;
    }

    const TImageInfo *CPaintManagerUI::AddImage(LPCTSTR bitmap, LPCTSTR type, DWORD mask, bool bUseHSL, bool bShared, HINSTANCE instance) {
        if( bitmap == NULL || bitmap[0] == _T('\0') )
            return NULL;

        TImageInfo *data = NULL;

        if( type != NULL && lstrlen(type) > 0) {
            if( isdigit(*bitmap) ) {
                LPTSTR pstr = NULL;
                int iIndex = _tcstol(bitmap, &pstr, 10);
                data = CRenderEngine::LoadImage(iIndex, type, mask, instance);
            }
        } else {
            data = CRenderEngine::LoadImage(bitmap, NULL, mask, instance);
        }

        if( data == NULL ) {
            return NULL;
        }

        data->bUseHSL = bUseHSL;

        if( type != NULL )
            data->sResType = type;

        data->dwMask = mask;

        if( data->bUseHSL ) {
            data->pSrcBits = new BYTE[data->nX * data->nY * 4];
            ::CopyMemory(data->pSrcBits, data->pBits, data->nX * data->nY * 4);
        } else {
            data->pSrcBits = NULL;
        }

        if( m_bUseHSL )
            CRenderEngine::AdjustImage(true, data, m_H, m_S, m_L);

        if (data) {
            if (bShared || m_bForceUseSharedRes) {
                TImageInfo *pOldImageInfo = static_cast<TImageInfo *>(m_SharedResInfo.m_ImageHash.Find(bitmap));

                if (pOldImageInfo) {
                    CRenderEngine::FreeImage(pOldImageInfo);
                    m_SharedResInfo.m_ImageHash.Remove(bitmap);
                }

                if( !m_SharedResInfo.m_ImageHash.Insert(bitmap, data) ) {
                    CRenderEngine::FreeImage(data);
                    data = NULL;
                }
            } else {
                TImageInfo *pOldImageInfo = static_cast<TImageInfo *>(m_ResInfo.m_ImageHash.Find(bitmap));

                if (pOldImageInfo) {
                    CRenderEngine::FreeImage(pOldImageInfo);
                    m_ResInfo.m_ImageHash.Remove(bitmap);
                }

                if( !m_ResInfo.m_ImageHash.Insert(bitmap, data) ) {
                    CRenderEngine::FreeImage(data);
                    data = NULL;
                }
            }
        }

        return data;
    }

    const TImageInfo *CPaintManagerUI::AddImage(LPCTSTR bitmap, HBITMAP hBitmap, int iWidth, int iHeight, bool bAlpha, bool bShared) {
        // 因无法确定外部HBITMAP格式，不能使用hsl调整
        if( bitmap == NULL || bitmap[0] == _T('\0') ) return NULL;

        if( hBitmap == NULL || iWidth <= 0 || iHeight <= 0 ) return NULL;

        TImageInfo *data = new TImageInfo;
        data->pBits = NULL;
        data->pSrcBits = NULL;
        data->hBitmap = hBitmap;
        data->pBits = NULL;
        data->nX = iWidth;
        data->nY = iHeight;
        data->bAlpha = bAlpha;
        data->bUseHSL = false;
        data->pSrcBits = NULL;
        data->dwMask = 0;

        if (bShared || m_bForceUseSharedRes) {
            if( !m_SharedResInfo.m_ImageHash.Insert(bitmap, data) ) {
                CRenderEngine::FreeImage(data);
                data = NULL;
            }
        } else {
            if( !m_ResInfo.m_ImageHash.Insert(bitmap, data) ) {
                CRenderEngine::FreeImage(data);
                data = NULL;
            }
        }

        return data;
    }

    void CPaintManagerUI::RemoveImage(LPCTSTR bitmap, bool bShared) {
        TImageInfo *data = NULL;

        if (bShared) {
            data = static_cast<TImageInfo *>(m_SharedResInfo.m_ImageHash.Find(bitmap));

            if (data) {
                CRenderEngine::FreeImage(data) ;
                m_SharedResInfo.m_ImageHash.Remove(bitmap);
            }
        } else {
            data = static_cast<TImageInfo *>(m_ResInfo.m_ImageHash.Find(bitmap));

            if (data) {
                CRenderEngine::FreeImage(data) ;
                m_ResInfo.m_ImageHash.Remove(bitmap);
            }
        }
    }

    void CPaintManagerUI::RemoveAllImages(bool bShared) {
        if (bShared) {
            TImageInfo *data = NULL;

            for (std::map<CDuiString, LPVOID>::iterator it = m_SharedResInfo.m_ImageHash.Begin(); it != m_SharedResInfo.m_ImageHash.End(); it++) {
                data = static_cast<TImageInfo *>(it->second);
                if (data) {
                    CRenderEngine::FreeImage(data);
                }
            }
            m_SharedResInfo.m_ImageHash.RemoveAll();
        } else {
            TImageInfo *data = NULL;
            for (std::map<CDuiString, LPVOID>::iterator it = m_ResInfo.m_ImageHash.Begin(); it != m_ResInfo.m_ImageHash.End(); it++) {
                data = static_cast<TImageInfo *>(it->second);
                if (data) {
                    CRenderEngine::FreeImage(data);
                }
            }
            m_ResInfo.m_ImageHash.RemoveAll();
        }
    }

    void CPaintManagerUI::AdjustSharedImagesHSL() {
        TImageInfo *data = NULL;
        for (std::map<CDuiString, LPVOID>::iterator it = m_SharedResInfo.m_ImageHash.Begin(); it != m_SharedResInfo.m_ImageHash.End(); it++) {
            data = static_cast<TImageInfo *>(it->second);

            if (data && data->bUseHSL) {
                CRenderEngine::AdjustImage(m_bUseHSL, data, m_H, m_S, m_L);
            }
        }
    }

    void CPaintManagerUI::AdjustImagesHSL() {
        TImageInfo *data = NULL;
        for (std::map<CDuiString, LPVOID>::iterator it = m_ResInfo.m_ImageHash.Begin(); it != m_ResInfo.m_ImageHash.End(); it++) {
            data = static_cast<TImageInfo *>(it->second);

            if (data && data->bUseHSL) {
                CRenderEngine::AdjustImage(m_bUseHSL, data, m_H, m_S, m_L);
            }
        }

        Invalidate();
    }

    void CPaintManagerUI::PostAsyncNotify() {
        if (!m_bAsyncNotifyPosted) {
            ::PostMessage(m_hWndPaint, WM_APP + 1, 0, 0L);
            m_bAsyncNotifyPosted = true;
        }
    }

    void CPaintManagerUI::ReloadSharedImages() {
        TImageInfo *data;
        TImageInfo *pNewData = NULL;

        for (std::map<CDuiString, LPVOID>::iterator it = m_SharedResInfo.m_ImageHash.Begin(); it != m_SharedResInfo.m_ImageHash.End(); it++) {
            data = static_cast<TImageInfo *>(it->second);

            if (data != NULL) {
                if (!data->sResType.IsEmpty()) {
                    if (isdigit(*(it->first))) {
                        LPTSTR pstr = NULL;
                        int iIndex = _tcstol(it->first, &pstr, 10);
                        pNewData = CRenderEngine::LoadImage(iIndex, data->sResType, data->dwMask);
                    }
                } else {
                    pNewData = CRenderEngine::LoadImage(it->first, NULL, data->dwMask);
                }

                if (pNewData == NULL)
                    continue;

                CRenderEngine::FreeImage(data, false);
                data->hBitmap = pNewData->hBitmap;
                data->pBits = pNewData->pBits;
                data->nX = pNewData->nX;
                data->nY = pNewData->nY;
                data->bAlpha = pNewData->bAlpha;
                data->pSrcBits = NULL;

                if (data->bUseHSL) {
                    data->pSrcBits = new BYTE[data->nX * data->nY * 4];
                    ::CopyMemory(data->pSrcBits, data->pBits, data->nX * data->nY * 4);
                } else {
                    data->pSrcBits = NULL;
                }

                if (m_bUseHSL)
                    CRenderEngine::AdjustImage(true, data, m_H, m_S, m_L);

                delete pNewData;
                pNewData = NULL;
            }
        }
    }

    void CPaintManagerUI::ReloadImages() {
        RemoveAllDrawInfos();

        TImageInfo *data = NULL;
        TImageInfo *pNewData = NULL;

        for (std::map<CDuiString, LPVOID>::iterator it = m_ResInfo.m_ImageHash.Begin(); it != m_ResInfo.m_ImageHash.End(); it++) {
            data = static_cast<TImageInfo *>(it->second);

            if (data != NULL) {
                if (!data->sResType.IsEmpty()) {
                    if (isdigit(*(it->first))) {
                        LPTSTR pstr = NULL;
                        int iIndex = _tcstol(it->first, &pstr, 10);
                        pNewData = CRenderEngine::LoadImage(iIndex, data->sResType, data->dwMask);
                    }
                } else {
                    pNewData = CRenderEngine::LoadImage(it->first, NULL, data->dwMask);
                }

                CRenderEngine::FreeImage(data, false);

                if (pNewData == NULL) {
                    m_ResInfo.m_ImageHash.Remove(it->first);
                    continue;
                }

                data->hBitmap = pNewData->hBitmap;
                data->pBits = pNewData->pBits;
                data->nX = pNewData->nX;
                data->nY = pNewData->nY;
                data->bAlpha = pNewData->bAlpha;
                data->pSrcBits = NULL;

                if (data->bUseHSL) {
                    data->pSrcBits = new BYTE[data->nX * data->nY * 4];
                    ::CopyMemory(data->pSrcBits, data->pBits, data->nX * data->nY * 4);
                } else
                    data->pSrcBits = NULL;

                if (m_bUseHSL)
                    CRenderEngine::AdjustImage(true, data, m_H, m_S, m_L);

                delete pNewData;
                pNewData = NULL;
            }
        }

        if( m_pRoot )
            m_pRoot->Invalidate();
    }

    const TDrawInfo *CPaintManagerUI::GetDrawInfo(LPCTSTR pStrImage, LPCTSTR pStrModify) {
        CDuiString sStrImage = pStrImage;
        CDuiString sStrModify = pStrModify;
        CDuiString sKey = sStrImage + sStrModify;
        TDrawInfo *pDrawInfo = static_cast<TDrawInfo *>(m_ResInfo.m_DrawInfoHash.Find(sKey));

        if(pDrawInfo == NULL && !sKey.IsEmpty()) {
            pDrawInfo = new TDrawInfo();
            pDrawInfo->Parse(pStrImage, pStrModify, this);
            m_ResInfo.m_DrawInfoHash.Insert(sKey, pDrawInfo);
        }

        return pDrawInfo;
    }

    void CPaintManagerUI::RemoveDrawInfo(LPCTSTR pStrImage, LPCTSTR pStrModify) {
        CDuiString sStrImage = pStrImage;
        CDuiString sStrModify = pStrModify;
        CDuiString sKey = sStrImage + sStrModify;
        TDrawInfo *pDrawInfo = static_cast<TDrawInfo *>(m_ResInfo.m_DrawInfoHash.Find(sKey));

        if(pDrawInfo != NULL) {
            m_ResInfo.m_DrawInfoHash.Remove(sKey);
            delete pDrawInfo;
            pDrawInfo = NULL;
        }
    }

    void CPaintManagerUI::RemoveAllDrawInfos() {
        TDrawInfo *pDrawInfo = NULL;
        for (std::map<CDuiString, LPVOID>::iterator it = m_ResInfo.m_DrawInfoHash.Begin(); it != m_ResInfo.m_DrawInfoHash.End(); it++) {
            pDrawInfo = static_cast<TDrawInfo *>(it->second);

            if (pDrawInfo) {
                delete pDrawInfo;
                pDrawInfo = NULL;
            }
        }
        m_ResInfo.m_DrawInfoHash.RemoveAll();
    }

    void CPaintManagerUI::AddDefaultAttributeList(LPCTSTR pStrControlName, LPCTSTR pStrControlAttrList, bool bShared) {
        if (bShared || m_bForceUseSharedRes) {
            CDuiString *pDefaultAttr = new CDuiString(pStrControlAttrList);

            if (pDefaultAttr != NULL) {
                CDuiString *pOldDefaultAttr = static_cast<CDuiString *>(m_SharedResInfo.m_AttrHash.Set(pStrControlName, (LPVOID)pDefaultAttr));

                if (pOldDefaultAttr)
                    delete pOldDefaultAttr;
            }
        } else {
            CDuiString *pDefaultAttr = new CDuiString(pStrControlAttrList);

            if (pDefaultAttr != NULL) {
                CDuiString *pOldDefaultAttr = static_cast<CDuiString *>(m_ResInfo.m_AttrHash.Set(pStrControlName, (LPVOID)pDefaultAttr));

                if (pOldDefaultAttr)
                    delete pOldDefaultAttr;
            }
        }
    }

    LPCTSTR CPaintManagerUI::GetDefaultAttributeList(LPCTSTR pStrControlName) const {
        CDuiString *pDefaultAttr = static_cast<CDuiString *>(m_ResInfo.m_AttrHash.Find(pStrControlName));

        if( !pDefaultAttr )
            pDefaultAttr = static_cast<CDuiString *>(m_SharedResInfo.m_AttrHash.Find(pStrControlName));

        if (pDefaultAttr)
            return pDefaultAttr->GetData();

        return NULL;
    }

    bool CPaintManagerUI::RemoveDefaultAttributeList(LPCTSTR pStrControlName, bool bShared) {
        if (bShared) {
            CDuiString *pDefaultAttr = static_cast<CDuiString *>(m_SharedResInfo.m_AttrHash.Find(pStrControlName));

            if( !pDefaultAttr )
                return false;

            delete pDefaultAttr;
            return m_SharedResInfo.m_AttrHash.Remove(pStrControlName);
        } else {
            CDuiString *pDefaultAttr = static_cast<CDuiString *>(m_ResInfo.m_AttrHash.Find(pStrControlName));

            if( !pDefaultAttr )
                return false;

            delete pDefaultAttr;
            return m_ResInfo.m_AttrHash.Remove(pStrControlName);
        }
    }

    void CPaintManagerUI::RemoveAllDefaultAttributeList(bool bShared) {
        if (bShared) {
            CDuiString *pDefaultAttr;
            for (std::map<CDuiString, LPVOID>::iterator it = m_SharedResInfo.m_AttrHash.Begin(); it != m_SharedResInfo.m_AttrHash.End(); it++) {
                pDefaultAttr = static_cast<CDuiString *>(it->second);

                if (pDefaultAttr)
                    delete pDefaultAttr;
            }
            m_SharedResInfo.m_AttrHash.RemoveAll();
        } else {
            CDuiString *pDefaultAttr;
            for (std::map<CDuiString, LPVOID>::iterator it = m_ResInfo.m_AttrHash.Begin(); it != m_ResInfo.m_AttrHash.End(); it++) {
                pDefaultAttr = static_cast<CDuiString *>(it->second);

                if (pDefaultAttr)
                    delete pDefaultAttr;
            }
            m_ResInfo.m_AttrHash.RemoveAll();
        }
    }

    void CPaintManagerUI::AddWindowCustomAttribute(LPCTSTR pstrName, LPCTSTR pstrAttr) {
        if( pstrName == NULL || pstrName[0] == _T('\0') || pstrAttr == NULL || pstrAttr[0] == _T('\0') )
            return;

        CDuiString *pCostomAttr = new CDuiString(pstrAttr);

        if (pCostomAttr != NULL) {
            if (m_mWindowCustomAttrHash.Find(pstrName) == NULL)
                m_mWindowCustomAttrHash.Set(pstrName, (LPVOID)pCostomAttr);
            else
                delete pCostomAttr;
        }
    }

    LPCTSTR CPaintManagerUI::GetWindowCustomAttribute(LPCTSTR pstrName) const {
        if( pstrName == NULL || pstrName[0] == _T('\0') )
            return NULL;

        CDuiString *pCostomAttr = static_cast<CDuiString *>(m_mWindowCustomAttrHash.Find(pstrName));

        if( pCostomAttr )
            return pCostomAttr->GetData();

        return NULL;
    }

    bool CPaintManagerUI::RemoveWindowCustomAttribute(LPCTSTR pstrName) {
        if( pstrName == NULL || pstrName[0] == _T('\0') )
            return NULL;

        CDuiString *pCostomAttr = static_cast<CDuiString *>(m_mWindowCustomAttrHash.Find(pstrName));

        if( !pCostomAttr )
            return false;

        delete pCostomAttr;
        return m_mWindowCustomAttrHash.Remove(pstrName);
    }

    void CPaintManagerUI::RemoveAllWindowCustomAttribute() {
        CDuiString *pCostomAttr;
        for (std::map<CDuiString, LPVOID>::iterator it = m_mWindowCustomAttrHash.Begin(); it != m_mWindowCustomAttrHash.End(); it++) {
            pCostomAttr = static_cast<CDuiString *>(it->second);
            if(pCostomAttr)
                delete pCostomAttr;
        }
        m_mWindowCustomAttrHash.RemoveAll();
    }

    CControlUI *CPaintManagerUI::GetRoot() const {
        ASSERT(m_pRoot);
        return m_pRoot;
    }

    CControlUI *CPaintManagerUI::FindControl(POINT pt) const {
        ASSERT(m_pRoot);
        return m_pRoot->FindControl(__FindControlFromPoint, &pt, UIFIND_VISIBLE | UIFIND_HITTEST | UIFIND_TOP_FIRST);
    }

    CControlUI *CPaintManagerUI::FindControl(LPCTSTR pstrName) const {
        ASSERT(m_pRoot);
        return static_cast<CControlUI *>(m_mNameHash.Find(pstrName));
    }

    CControlUI *CPaintManagerUI::FindSubControlByPoint(CControlUI *pParent, POINT pt) const {
        if( pParent == NULL )
            pParent = GetRoot();

        ASSERT(pParent);
        return pParent->FindControl(__FindControlFromPoint, &pt, UIFIND_VISIBLE | UIFIND_HITTEST | UIFIND_TOP_FIRST);
    }

    CControlUI *CPaintManagerUI::FindSubControlByName(CControlUI *pParent, LPCTSTR pstrName) const {
        if( pParent == NULL )
            pParent = GetRoot();

        ASSERT(pParent);
        return pParent->FindControl(__FindControlFromName, (LPVOID)pstrName, UIFIND_ALL);
    }

    CControlUI *CPaintManagerUI::FindSubControlByClass(CControlUI *pParent, LPCTSTR pstrClass, int iIndex) {
        if( pParent == NULL )
            pParent = GetRoot();

        ASSERT(pParent);
        m_aFoundControls.Resize(iIndex + 1);
        return pParent->FindControl(__FindControlFromClass, (LPVOID)pstrClass, UIFIND_ALL);
    }

    CStdPtrArray *CPaintManagerUI::FindSubControlsByClass(CControlUI *pParent, LPCTSTR pstrClass) {
        if( pParent == NULL )
            pParent = GetRoot();

        ASSERT(pParent);
        m_aFoundControls.Empty();
        pParent->FindControl(__FindControlsFromClass, (LPVOID)pstrClass, UIFIND_ALL);
        return &m_aFoundControls;
    }

    CStdPtrArray *CPaintManagerUI::GetFoundControls() {
        return &m_aFoundControls;
    }

    CControlUI *CALLBACK CPaintManagerUI::__FindControlFromNameHash(CControlUI *pThis, LPVOID pData) {
        CPaintManagerUI *pManager = static_cast<CPaintManagerUI *>(pData);
        const CDuiString &sName = pThis->GetName();

        if( sName.IsEmpty() )
            return NULL;

        // Add this control to the hash list
        pManager->m_mNameHash.Set(sName, pThis);
        return NULL; // Attempt to add all controls
    }

    CControlUI *CALLBACK CPaintManagerUI::__FindControlFromCount(CControlUI * /*pThis*/, LPVOID pData) {
        int *pnCount = static_cast<int *>(pData);
        (*pnCount)++;
        return NULL;  // Count all controls
    }

    CControlUI *CALLBACK CPaintManagerUI::__FindControlFromPoint(CControlUI *pThis, LPVOID pData) {
        LPPOINT pPoint = static_cast<LPPOINT>(pData);
        return ::PtInRect(&pThis->GetPos(), *pPoint) ? pThis : NULL;
    }

    CControlUI *CALLBACK CPaintManagerUI::__FindControlFromTab(CControlUI *pThis, LPVOID pData) {
        FINDTABINFO *pInfo = static_cast<FINDTABINFO *>(pData);

        if( pInfo->pFocus == pThis ) {
            if( pInfo->bForward ) pInfo->bNextIsIt = true;

            return pInfo->bForward ? NULL : pInfo->pLast;
        }

        if( (pThis->GetControlFlags() & UIFLAG_TABSTOP) == 0 ) return NULL;

        pInfo->pLast = pThis;

        if( pInfo->bNextIsIt ) return pThis;

        if( pInfo->pFocus == NULL ) return pThis;

        return NULL;  // Examine all controls
    }

    CControlUI *CALLBACK CPaintManagerUI::__FindControlFromShortcut(CControlUI *pThis, LPVOID pData) {
        if( !pThis->IsVisible() )
            return NULL;

        FINDSHORTCUT *pFS = static_cast<FINDSHORTCUT *>(pData);

        if( pFS->ch == toupper(pThis->GetShortcut()) )
            pFS->bPickNext = true;

        if( _tcsstr(pThis->GetClass(), DUI_CTR_LABEL) != NULL )
            return NULL;   // Labels never get focus!

        return pFS->bPickNext ? pThis : NULL;
    }

    CControlUI *CALLBACK CPaintManagerUI::__FindControlFromName(CControlUI *pThis, LPVOID pData) {
        LPCTSTR pstrName = static_cast<LPCTSTR>(pData);
        CDuiString sName = pThis->GetName();

        if( sName.IsEmpty() )
            return NULL;

        return (_tcsicmp(sName, pstrName) == 0) ? pThis : NULL;
    }

    CControlUI *CALLBACK CPaintManagerUI::__FindControlFromClass(CControlUI *pThis, LPVOID pData) {
        LPCTSTR pstrType = static_cast<LPCTSTR>(pData);
        LPCTSTR pType = pThis->GetClass();
        CStdPtrArray *pFoundControls = pThis->GetManager()->GetFoundControls();

        if( _tcscmp(pstrType, _T("*")) == 0 || _tcscmp(pstrType, pType) == 0 ) {
            int iIndex = -1;

            while( pFoundControls->GetAt(++iIndex) != NULL ) ;

            if( iIndex < pFoundControls->GetSize() )
                pFoundControls->SetAt(iIndex, pThis);
        }

        if( pFoundControls->GetAt(pFoundControls->GetSize() - 1) != NULL )
            return pThis;

        return NULL;
    }

    CControlUI *CALLBACK CPaintManagerUI::__FindControlsFromClass(CControlUI *pThis, LPVOID pData) {
        LPCTSTR pstrType = static_cast<LPCTSTR>(pData);
        LPCTSTR pType = pThis->GetClass();

        if( _tcscmp(pstrType, _T("*")) == 0 || _tcscmp(pstrType, pType) == 0 )
            pThis->GetManager()->GetFoundControls()->Add((LPVOID)pThis);

        return NULL;
    }

    CControlUI *CALLBACK CPaintManagerUI::__FindControlsFromUpdate(CControlUI *pThis, LPVOID pData) {
        if( pThis->IsUpdateNeeded() ) {
            pThis->GetManager()->GetFoundControls()->Add((LPVOID)pThis);
            return pThis;
        }

        return NULL;
    }

    bool CPaintManagerUI::TranslateAccelerator(LPMSG pMsg) {
        for (int i = 0; i < m_aTranslateAccelerator.GetSize(); i++) {
            LRESULT lResult = static_cast<ITranslateAccelerator *>(m_aTranslateAccelerator[i])->TranslateAccelerator(pMsg);

            if( lResult == S_OK )
                return true;
        }

        return false;
    }

    bool CPaintManagerUI::TranslateMessage(const LPMSG pMsg) {
        // Pretranslate Message takes care of system-wide messages, such as
        // tabbing and shortcut key-combos. We'll look for all messages for
        // each window and any child control attached.
        UINT uStyle = GetWindowStyle(pMsg->hwnd);
        UINT uChildRes = uStyle & WS_CHILD;
        LRESULT lRes = 0;

        if (uChildRes != 0) {
            HWND hWndParent = ::GetParent(pMsg->hwnd);

            for( int i = 0; i < m_aPreMessages.GetSize(); i++ ) {
                CPaintManagerUI *pT = static_cast<CPaintManagerUI *>(m_aPreMessages[i]);
                HWND hTempParent = hWndParent;

                while(hTempParent) {
                    if(pMsg->hwnd == pT->GetPaintWindow() || hTempParent == pT->GetPaintWindow()) {
                        if (pT->TranslateAccelerator(pMsg))
                            return true;

                        pT->PreMessageHandler(pMsg->message, pMsg->wParam, pMsg->lParam, lRes);
                    }

                    hTempParent = GetParent(hTempParent);
                }
            }
        } else {
            for( int i = 0; i < m_aPreMessages.GetSize(); i++ ) {
                CPaintManagerUI *pT = static_cast<CPaintManagerUI *>(m_aPreMessages[i]);

                if(pMsg->hwnd == pT->GetPaintWindow()) {
                    if (pT->TranslateAccelerator(pMsg))
                        return true;

                    if( pT->PreMessageHandler(pMsg->message, pMsg->wParam, pMsg->lParam, lRes) )
                        return true;

                    return false;
                }
            }
        }

        return false;
    }

    bool CPaintManagerUI::AddTranslateAccelerator(ITranslateAccelerator *pTranslateAccelerator) {
        ASSERT(m_aTranslateAccelerator.Find(pTranslateAccelerator) < 0);
        return m_aTranslateAccelerator.Add(pTranslateAccelerator);
    }

    bool CPaintManagerUI::RemoveTranslateAccelerator(ITranslateAccelerator *pTranslateAccelerator) {
        for (int i = 0; i < m_aTranslateAccelerator.GetSize(); i++) {
            if (static_cast<ITranslateAccelerator *>(m_aTranslateAccelerator[i]) == pTranslateAccelerator) {
                return m_aTranslateAccelerator.Remove(i);
            }
        }

        return false;
    }

    void CPaintManagerUI::UsedVirtualWnd(bool bUsed) {
        m_bUsedVirtualWnd = bUsed;
    }

    // 样式管理
    void CPaintManagerUI::AddStyle(LPCTSTR pName, LPCTSTR pDeclarationList, bool bShared) {
        CDuiString *pStyle = new CDuiString(pDeclarationList);

        if(bShared || m_bForceUseSharedRes) {
            if( !m_SharedResInfo.m_StyleHash.Insert(pName, pStyle) ) {
                delete pStyle;
            }
        } else {
            if( !m_ResInfo.m_StyleHash.Insert(pName, pStyle) ) {
                delete pStyle;
            }
        }
    }

    LPCTSTR CPaintManagerUI::GetStyle(LPCTSTR pName) const {
        CDuiString *pStyle = static_cast<CDuiString *>(m_ResInfo.m_StyleHash.Find(pName));

        if( !pStyle )
            pStyle = static_cast<CDuiString *>(m_SharedResInfo.m_StyleHash.Find(pName));

        if( pStyle )
            return pStyle->GetData();
        else return NULL;
    }

    BOOL CPaintManagerUI::RemoveStyle(LPCTSTR pName, bool bShared) {
        CDuiString *pStyle = NULL;

        if (bShared) {
            pStyle = static_cast<CDuiString *>(m_SharedResInfo.m_StyleHash.Find(pName));

            if (pStyle) {
                delete pStyle;
                m_SharedResInfo.m_StyleHash.Remove(pName);
            }
        } else {
            pStyle = static_cast<CDuiString *>(m_ResInfo.m_StyleHash.Find(pName));

            if (pStyle) {
                delete pStyle;
                m_ResInfo.m_StyleHash.Remove(pName);
            }
        }

        return true;
    }

    const CStdStringPtrMap &CPaintManagerUI::GetStyles(bool bShared) const {
        if(bShared)
            return m_SharedResInfo.m_StyleHash;
        else
            return m_ResInfo.m_StyleHash;
    }

    void CPaintManagerUI::RemoveAllStyle(bool bShared) {
        if (bShared) {
            CDuiString *pStyle = NULL;
            for (std::map<CDuiString, LPVOID>::iterator it = m_SharedResInfo.m_StyleHash.Begin(); it != m_SharedResInfo.m_StyleHash.End(); it++) {
                pStyle = static_cast<CDuiString *>(it->second);
                if(pStyle)
                    delete pStyle;
            }
            m_SharedResInfo.m_StyleHash.RemoveAll();
        } else {
            CDuiString *pStyle = NULL;
            for (std::map<CDuiString, LPVOID>::iterator it = m_ResInfo.m_StyleHash.Begin(); it != m_ResInfo.m_StyleHash.End(); it++) {
                pStyle = static_cast<CDuiString *>(it->second);
                if (pStyle)
                    delete pStyle;
            }
            m_ResInfo.m_StyleHash.RemoveAll();
        }
    }

    const TImageInfo *CPaintManagerUI::GetImageString(LPCTSTR pStrImage, LPCTSTR pStrModify) {
        CDuiString sImageName = pStrImage;
        CDuiString sImageResType = _T("");
        DWORD dwMask = 0;
        CDuiString sItem;
        CDuiString sValue;
        LPTSTR pstr = NULL;

        for( int i = 0; i < 2; ++i) {
            if( i == 1)
                pStrImage = pStrModify;

            if( !pStrImage )
                continue;

            while( *pStrImage != _T('\0') ) {
                sItem.Empty();
                sValue.Empty();

                while( *pStrImage > _T('\0') && *pStrImage <= _T(' ') ) pStrImage = ::CharNext(pStrImage);

                while( *pStrImage != _T('\0') && *pStrImage != _T('=') && *pStrImage > _T(' ') ) {
                    LPTSTR pstrTemp = ::CharNext(pStrImage);

                    while( pStrImage < pstrTemp) {
                        sItem += *pStrImage++;
                    }
                }

                while( *pStrImage > _T('\0') && *pStrImage <= _T(' ') ) pStrImage = ::CharNext(pStrImage);

                if( *pStrImage++ != _T('=') )
                    break;

                while( *pStrImage > _T('\0') && *pStrImage <= _T(' ') ) pStrImage = ::CharNext(pStrImage);

                if( *pStrImage++ != _T('\'') )
                    break;

                while( *pStrImage != _T('\0') && *pStrImage != _T('\'') ) {
                    LPTSTR pstrTemp = ::CharNext(pStrImage);

                    while( pStrImage < pstrTemp) {
                        sValue += *pStrImage++;
                    }
                }

                if( *pStrImage++ != _T('\'') )
                    break;

                if( !sValue.IsEmpty() ) {
                    if( sItem == _T("file") || sItem == _T("res") ) {
                        sImageName = sValue;
                    } else if( sItem == _T("restype") ) {
                        sImageResType = sValue;
                    } else if( sItem == _T("mask") ) {
                        if( sValue[0] == _T('#')) dwMask = _tcstoul(sValue.GetData() + 1, &pstr, 16);
                        else dwMask = _tcstoul(sValue, &pstr, 16);
                    }

                }

                if( *pStrImage++ != _T(' ') )
                    break;
            }
        }

        return GetImageEx(sImageName, sImageResType, dwMask);
    }

    static WORD DIBNumColors(void *pv) {
        int bits;
        LPBITMAPINFOHEADER  lpbi;
        LPBITMAPCOREHEADER  lpbc;
        lpbi = ((LPBITMAPINFOHEADER)pv);
        lpbc = ((LPBITMAPCOREHEADER)pv);

        /*  With the BITMAPINFO format headers, the size of the palette
        *  is in biClrUsed, whereas in the BITMAPCORE - style headers, it
        *  is dependent on the bits per pixel ( = 2 raised to the power of
        *  bits/pixel).
        */
        if (lpbi->biSize != sizeof(BITMAPCOREHEADER)) {
            if (lpbi->biClrUsed != 0)
                return (WORD)lpbi->biClrUsed;

            bits = lpbi->biBitCount;
        } else
            bits = lpbc->bcBitCount;

        switch (bits) {
            case 1:
                return 2;

            case 4:
                return 16;

            case 8:
                return 256;

            default:
                /* A 24 bitcount DIB has no color table */
                return 0;
        }
    }
    //code taken from SEEDIB MSDN sample
    static WORD ColorTableSize(LPVOID lpv) {
        LPBITMAPINFOHEADER lpbih = (LPBITMAPINFOHEADER)lpv;

        if (lpbih->biSize != sizeof(BITMAPCOREHEADER)) {
            if (((LPBITMAPINFOHEADER)(lpbih))->biCompression == BI_BITFIELDS)
                /* Remember that 16/32bpp dibs can still have a color table */
                return (sizeof(DWORD) * 3) + (DIBNumColors (lpbih) * sizeof (RGBQUAD));
            else
                return (WORD)(DIBNumColors (lpbih) * sizeof (RGBQUAD));
        } else
            return (WORD)(DIBNumColors (lpbih) * sizeof (RGBTRIPLE));
    }
} // namespace DuiLib
